2 * Copyright (c) 2000-2006 Apple Computer, Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
29 * Copyright (c) 1995 Apple Computer, Inc.
32 * Created, March 17, 1997 by Tuyen Nguyen for MacOSX.
35 #include <sys/errno.h>
36 #include <sys/types.h>
37 #include <sys/param.h>
38 #include <machine/spl.h>
39 #include <sys/systm.h>
40 #include <sys/kernel.h>
42 #include <sys/filedesc.h>
43 #include <sys/fcntl.h>
44 #include <sys/file_internal.h>
46 #include <sys/ioctl.h>
47 #include <sys/malloc.h>
48 #include <kern/locks.h>
49 #include <sys/socket.h>
50 #include <sys/socketvar.h>
51 #include <sys/ioccom.h>
52 #include <sys/uio_internal.h>
54 #include <sys/vnode.h>
56 #include <sys/sysctl.h>
60 #include <netat/sysglue.h>
61 #include <netat/appletalk.h>
62 #include <netat/ddp.h>
63 #include <netat/at_pcb.h>
64 #include <netat/at_var.h>
65 #include <netat/routing_tables.h>
66 #include <netat/adsp.h>
67 #include <netat/adsp_internal.h>
68 #include <netat/asp.h>
69 #include <netat/atp.h>
70 #include <netat/debug.h>
72 int _ATkqfilter(struct fileproc
*, struct knote
*, vfs_context_t
);
73 int _ATselect(struct fileproc
*, int, void *, vfs_context_t
);
74 int _ATioctl(struct fileproc
*, u_long
, caddr_t
, vfs_context_t
);
75 int _ATwrite(struct fileproc
*, struct uio
*, int, vfs_context_t
);
76 int _ATread(struct fileproc
*, struct uio
*, int, vfs_context_t
);
77 int _ATclose(struct fileglob
*, vfs_context_t
);
79 int _ATrw(struct fileproc
*, enum uio_rw
, struct uio
*, vfs_context_t
);
81 extern struct atpcb ddp_head
;
82 extern lck_mtx_t
* atalk_mutex
;
84 int atp_free_cluster_timeout_set
= 0;
86 int gref_alloc(gref_t
**);
89 /* bms: make gref_close non static so its callable from kernel */
90 int gref_close(gref_t
*gref
);
92 SYSCTL_DECL(_net_appletalk
);
94 SYSCTL_STRUCT(_net_appletalk
, OID_AUTO
, debug
, CTLFLAG_WR
,
95 &dbgBits
, dbgBits
, "AppleTalk Debug Flags");
96 int RouterMix
= RT_MIX_DEFAULT
; /* default for nbr of ppsec */
97 SYSCTL_INT(_net_appletalk
, OID_AUTO
, routermix
, CTLFLAG_WR
,
98 &RouterMix
, 0, "Appletalk RouterMix");
99 at_ddp_stats_t at_ddp_stats
; /* DDP statistics */
100 SYSCTL_STRUCT(_net_appletalk
, OID_AUTO
, ddpstats
, CTLFLAG_RD
| CTLFLAG_LOCKED
,
101 &at_ddp_stats
, at_ddp_stats
, "AppleTalk DDP Stats");
102 extern int atp_resp_seqno2big
;
103 SYSCTL_INT(_net_appletalk
, OID_AUTO
, atp_resp_seqno2big
, CTLFLAG_RD
| CTLFLAG_LOCKED
,
104 &atp_resp_seqno2big
, 0, "Appletalk ATP seqno too big count");
106 static void ioccmd_t_32_to_64( ioccmd_t
*from_p
, user_ioccmd_t
*to_p
);
107 static void ioccmd_t_64_to_32( user_ioccmd_t
*from_p
, ioccmd_t
*to_p
);
109 extern lck_mtx_t
*atalk_cluster_lock
;
110 caddr_t atp_free_cluster_list
= NULL
;
112 void gref_wput(gref_t
*, gbuf_t
*m
);
114 void gref_wput(gref
, m
)
118 switch (gref
->proto
) {
120 ddp_putmsg(gref
, m
); break;
122 elap_wput(gref
, m
); break;
124 atp_wput(gref
, m
); break;
126 asp_wput(gref
, m
); break;
129 aurp_wput(gref
, m
); break;
132 adsp_wput(gref
, m
); break;
134 if (gbuf_type(m
) == MSG_IOCTL
) {
135 gbuf_freem(gbuf_cont(m
));
137 ((ioc_t
*)gbuf_rptr(m
))->ioc_rval
= -1;
138 ((ioc_t
*)gbuf_rptr(m
))->ioc_error
= EPROTOTYPE
;
139 gbuf_set_type(m
, MSG_IOCNAK
);
140 atalk_putnext(gref
, m
);
150 int _ATsocket(proto
, err
, proc
)
158 /* make sure the specified protocol id is valid */
161 /* ATPROTO_DDP and ATPROTO_LAP have been replaced with
162 BSD-style socket interface. */
173 #ifdef APPLETALK_DEBUG
174 kprintf("_ATsocket: error EPROTOTYPE =%d\n", *err
);
179 /* allocate a protocol channel */
180 if ((*err
= gref_alloc(&gref
)) != 0) {
181 #ifdef APPLETALK_DEBUG
182 kprintf("_ATsocket: error gref_open =%d\n", *err
);
187 gref
->pid
= proc_pid((struct proc
*)proc
);
189 /* open the specified protocol */
190 switch (gref
->proto
) {
192 /* ATPROTO_DDP and ATPROTO_LAP have been replaced with
193 BSD-style socket interface. */
196 *err
= atp_open(gref
, 1); break;
198 *err
= asp_open(gref
); break;
201 *err
= aurp_open(gref
); break;
204 *err
= adsp_open(gref
); break;
207 /* create the descriptor for the channel */
209 #ifdef APPLETALK_DEBUG
210 kprintf("_ATsocket: open failed for %d proto; err = %d\n",
213 gref
->proto
= ATPROTO_NONE
;
215 if (*err
|| (*err
= atalk_openref(gref
, &fd
, proc
))) {
216 #ifdef APPLETALK_DEBUG
217 kprintf("_ATsocket: error atalk_openref =%d\n", *err
);
219 (void)gref_close(gref
);
223 kprintf("_ATsocket: proto=%d return=%d fd=%d\n", proto, *err, fd);
228 int _ATgetmsg(fd
, ctlptr
, datptr
, flags
, err
, proc
)
239 if ((*err
= atalk_getref(0, fd
, &gref
, proc
, 1)) == 0) {
240 switch (gref
->proto
) {
242 rc
= ASPgetmsg(gref
, ctlptr
, datptr
, NULL
, flags
, err
);
246 rc
= AURPgetmsg(err
);
250 *err
= EPROTONOSUPPORT
;
256 /* kprintf("_ATgetmsg: return=%d\n", *err);*/
260 int _ATputmsg(fd
, ctlptr
, datptr
, flags
, err
, proc
)
271 if ((*err
= atalk_getref(0, fd
, &gref
, proc
, 1)) == 0) {
272 switch (gref
->proto
) {
274 rc
= ASPputmsg(gref
, ctlptr
, datptr
, NULL
, flags
, err
); break;
276 *err
= EPROTONOSUPPORT
; break;
281 /* kprintf("_ATputmsg: return=%d\n", *err); */
287 __unused vfs_context_t ctx
)
292 if ((err
= atalk_closeref(fg
, &gref
)) == 0) {
294 (void)gref_close(gref
);
301 int _ATrw(fp
, rw
, uio
, ctx
)
307 int err
, len
, clen
= 0, res
;
309 gbuf_t
*m
, *mhead
, *mprev
;
310 proc_t p
= vfs_context_proc(ctx
);
312 /* no need to get/drop iocount as the fp already has one */
313 if ((err
= atalk_getref_locked(fp
, 0, &gref
, p
, 1)) != 0)
316 // LP64todo - fix this!
317 if ((len
= uio_resid(uio
)) == 0)
321 if (rw
== UIO_READ
) {
322 KERNEL_DEBUG(DBG_ADSP_ATRW
, 0, gref
, len
, gref
->rdhead
, 0);
323 while ((gref
->errno
== 0) && ((mhead
= gref
->rdhead
) == 0)) {
324 gref
->sevents
|= POLLMSG
;
325 err
= msleep(&gref
->event
, atalk_mutex
, PSOCK
| PCATCH
, "AT read", 0);
326 gref
->sevents
&= ~POLLMSG
;
329 KERNEL_DEBUG(DBG_ADSP_ATRW
, 1, gref
, gref
->rdhead
, mhead
, gbuf_next(mhead
));
334 if ((gref
->rdhead
= gbuf_next(mhead
)) == 0)
337 KERNEL_DEBUG(DBG_ADSP_ATRW
, 2, gref
, gref
->rdhead
, mhead
, gbuf_next(mhead
));
340 //##### LD TEST 08/05
341 // simple_lock(&gref->lock);
343 gbuf_next(mhead
) = 0;
345 for (mprev
=0, m
=mhead
; m
&& len
; len
-=clen
) {
346 if ((clen
= gbuf_len(m
)) > 0) {
349 uio
->uio_rw
= UIO_READ
;
350 if ((res
= uiomove((caddr_t
)gbuf_rptr(m
),
352 KERNEL_DEBUG(DBG_ADSP_ATRW
, 3, m
, clen
,
356 if (gbuf_len(m
) > len
) {
365 KERNEL_DEBUG(DBG_ADSP_ATRW
, 4, m
, gbuf_len(m
), mprev
, gref
->rdhead
);
367 gbuf_cont(mprev
) = 0;
370 if (gref
->rdhead
== 0)
372 gbuf_next(m
) = gref
->rdhead
;
378 // simple_unlock(&gref->lock);
380 if (gref
->writeable
) {
381 while (!(*gref
->writeable
)(gref
)) {
382 /* flow control on, wait to be enabled to write */
383 gref
->sevents
|= POLLSYNC
;
384 err
= msleep(&gref
->event
, atalk_mutex
, PSOCK
| PCATCH
, "AT write", 0);
385 gref
->sevents
&= ~POLLSYNC
;
392 /* allocate a buffer to copy in the write data */
393 if ((m
= gbuf_alloc(AT_WR_OFFSET
+len
, PRI_MED
)) == 0)
395 gbuf_rinc(m
,AT_WR_OFFSET
);
398 /* copy in the write data */
399 uio
->uio_rw
= UIO_WRITE
;
400 if ((res
= uiomove((caddr_t
)gbuf_rptr(m
), len
, uio
))) {
401 #ifdef APPLETALK_DEBUG
402 kprintf("_ATrw: UIO_WRITE: res=%d\n", res
);
408 /* forward the write data to the appropriate protocol module */
424 stat
= _ATrw(fp
, UIO_READ
, uio
, ctx
);
438 stat
= _ATrw(fp
, UIO_WRITE
, uio
, ctx
);
444 /* Most of the processing from _ATioctl, so that it can be called
445 from the new ioctl code */
446 /* bms: update to be callable from kernel */
447 int at_ioctl(gref_t
*gref
, u_long cmd
, caddr_t arg
, int fromKernel
)
453 user_addr_t user_arg
;
454 user_ioccmd_t user_ioccmd
;
457 /* error if not for us */
458 if ((cmd
& 0xffff) != 0xff99)
461 size
= IOCPARM_LEN(cmd
);
462 if (size
!= sizeof(user_addr_t
))
465 user_arg
= *((user_addr_t
*)arg
);
467 /* copy in ioc command info */
468 is64bit
= proc_is64bit(current_proc());
471 bcopy (CAST_DOWN(caddr_t
, user_arg
), &tmp
, sizeof (tmp
));
472 ioccmd_t_32_to_64(&tmp
, &user_ioccmd
);
476 err
= copyin(user_arg
, (caddr_t
)&user_ioccmd
, sizeof(user_ioccmd
));
480 err
= copyin(user_arg
, (caddr_t
)&tmp
, sizeof(tmp
));
481 ioccmd_t_32_to_64(&tmp
, &user_ioccmd
);
484 #ifdef APPLETALK_DEBUG
485 kprintf("at_ioctl: err = %d, copyin(%llx, %x, %d)\n", err
,
486 user_arg
, (caddr_t
)&user_ioccmd
, sizeof(user_ioccmd
));
492 /* allocate a buffer to create an ioc command
493 first mbuf contains ioc command */
494 if ((m
= gbuf_alloc(sizeof(ioc_t
), PRI_HI
)) == 0)
496 gbuf_wset(m
, sizeof(ioc_t
)); /* mbuf->m_len */
497 gbuf_set_type(m
, MSG_IOCTL
); /* mbuf->m_type */
499 /* create the ioc command
500 second mbuf contains the actual ASP command */
501 if (user_ioccmd
.ic_len
) {
502 if ((gbuf_cont(m
) = gbuf_alloc(user_ioccmd
.ic_len
, PRI_HI
)) == 0) {
504 #ifdef APPLETALK_DEBUG
505 kprintf("at_ioctl: gbuf_alloc err=%d\n",ENOBUFS
);
509 gbuf_wset(gbuf_cont(m
), user_ioccmd
.ic_len
); /* mbuf->m_len */
511 bcopy (CAST_DOWN(caddr_t
, user_ioccmd
.ic_dp
), gbuf_rptr(gbuf_cont(m
)), user_ioccmd
.ic_len
);
513 if ((err
= copyin(user_ioccmd
.ic_dp
, (caddr_t
)gbuf_rptr(gbuf_cont(m
)), user_ioccmd
.ic_len
)) != 0) {
519 ioc
= (ioc_t
*) gbuf_rptr(m
);
520 ioc
->ioc_cmd
= user_ioccmd
.ic_cmd
;
521 ioc
->ioc_count
= user_ioccmd
.ic_len
;
525 /* send the ioc command to the appropriate recipient */
528 /* wait for the ioc ack */
529 while ((m
= gref
->ichead
) == 0) {
530 gref
->sevents
|= POLLPRI
;
531 #ifdef APPLETALK_DEBUG
532 kprintf("sleep gref = 0x%x\n", (unsigned)gref
);
534 err
= msleep(&gref
->iocevent
, atalk_mutex
, PSOCK
| PCATCH
, "AT ioctl", 0);
535 gref
->sevents
&= ~POLLPRI
;
537 #ifdef APPLETALK_DEBUG
538 kprintf("at_ioctl: EINTR\n");
545 if (gbuf_next(m
) == m
) /* error case */
548 gref
->ichead
= gbuf_next(m
);
551 #ifdef APPLETALK_DEBUG
552 kprintf("at_ioctl: woke up from ioc sleep gref = 0x%x\n",
556 /* process the ioc response */
557 ioc
= (ioc_t
*) gbuf_rptr(m
);
558 if ((err
= ioc
->ioc_error
) == 0) {
559 user_ioccmd
.ic_timout
= ioc
->ioc_rval
;
560 user_ioccmd
.ic_len
= 0;
561 mdata
= gbuf_cont(m
);
562 if (mdata
&& user_ioccmd
.ic_dp
) {
563 user_ioccmd
.ic_len
= gbuf_msgsize(mdata
);
564 for (len
= 0; mdata
; mdata
= gbuf_cont(mdata
)) {
566 bcopy (gbuf_rptr(mdata
), CAST_DOWN(caddr_t
, (user_ioccmd
.ic_dp
+ len
)), gbuf_len(mdata
));
568 if ((err
= copyout((caddr_t
)gbuf_rptr(mdata
), (user_ioccmd
.ic_dp
+ len
), gbuf_len(mdata
))) < 0) {
569 #ifdef APPLETALK_DEBUG
570 kprintf("at_ioctl: len=%d error copyout=%d from=%x to=%x gbuf_len=%x\n",
571 len
, err
, (caddr_t
)gbuf_rptr(mdata
), (caddr_t
)&user_ioccmd
.ic_dp
[len
], gbuf_len(mdata
));
576 len
+= gbuf_len(mdata
);
582 ioccmd_t_64_to_32(&user_ioccmd
, &tmp
);
583 bcopy (&tmp
, CAST_DOWN(caddr_t
, user_arg
), sizeof(tmp
));
587 err
= copyout((caddr_t
)&user_ioccmd
, user_arg
, sizeof(user_ioccmd
));
591 ioccmd_t_64_to_32(&user_ioccmd
, &tmp
);
592 err
= copyout((caddr_t
)&tmp
, user_arg
, sizeof(tmp
));
602 /*kprintf("at_ioctl: I_done=%d\n", err);*/
609 register caddr_t arg
,
610 __unused vfs_context_t ctx
)
616 /* No need to get a reference on fp as it already has one */
617 if ((err
= atalk_getref_locked(fp
, 0, &gref
, 0, 0)) != 0) {
618 #ifdef APPLETALK_DEBUG
619 kprintf("_ATioctl: atalk_getref err = %d\n", err
);
623 err
= at_ioctl(gref
, cmd
, arg
, 0);
630 int _ATselect(fp
, which
, wql
, ctx
)
638 proc_t proc
= vfs_context_proc(ctx
);
640 /* Radar 4128949: Drop the proc_fd lock here to avoid lock inversion issues with the other AT calls
641 * select() is already holding a reference on the fd, so it won't go away during the time it is unlocked.
646 /* no need to drop the iocount as select covers that */
647 err
= atalk_getref_locked(fp
, 0, &gref
, 0, 0);
650 /* Safe to re-grab the proc_fdlock at that point */
655 if (which
== FREAD
) {
656 if (gref
->rdhead
|| (gref
->readable
&& (*gref
->readable
)(gref
)))
659 gref
->sevents
|= POLLIN
;
660 selrecord(proc
, &gref
->si
, wql
);
663 else if (which
== POLLOUT
) {
664 if (gref
->writeable
) {
665 if ((*gref
->writeable
)(gref
))
668 gref
->sevents
|= POLLOUT
;
669 selrecord(proc
, &gref
->si
, wql
);
680 __unused
struct fileproc
*fp
,
681 __unused
struct knote
*kn
,
682 __unused vfs_context_t ctx
)
687 void atalk_putnext(gref
, m
)
693 /* *** potential leak? *** */
696 switch (gbuf_type(m
)) {
700 gbuf_next(gref
->ichead
) = m
;
703 if (gref
->sevents
& POLLPRI
) {
704 #ifdef APPLETALK_DEBUG
705 kprintf("wakeup gref = 0x%x\n", (unsigned)gref
);
707 wakeup(&gref
->iocevent
);
712 /* *** this processing was moved to atalk_notify *** */
713 panic("atalk_putnext receved MSG_ERROR");
720 gbuf_next(gref
->rdtail
) = m
;
724 if (gref
->sevents
& POLLMSG
) {
725 gref
->sevents
&= ~POLLMSG
;
726 wakeup(&gref
->event
);
728 if (gref
->sevents
& POLLIN
) {
729 gref
->sevents
&= ~POLLIN
;
730 selwakeup(&gref
->si
);
734 } /* switch gbuf_type(m) */
736 } /* atalk_putnext */
738 void atalk_enablew(gref
)
741 if (gref
->sevents
& POLLSYNC
)
742 wakeup(&gref
->event
);
745 void atalk_flush(gref
)
750 gbuf_freel(gref
->rdhead
);
754 gbuf_freel(gref
->ichead
);
760 * Notify an appletalk user of an asynchronous error;
761 * just wake up so that he can collect error status.
763 void atalk_notify(gref
, errno
)
764 register gref_t
*gref
;
768 if (gref
->atpcb_socket
) {
770 This section is patterned after udp_notify() in
773 gref
->atpcb_socket
->so_error
= errno
;
774 sorwakeup(gref
->atpcb_socket
);
775 sowwakeup(gref
->atpcb_socket
);
777 /* for ATP, ASP, and ADSP */
778 if (gref
->errno
== 0) {
780 /* clear out data waiting to be read */
782 gbuf_freel(gref
->rdhead
);
786 if (gref
->sevents
& POLLMSG
) {
787 gref
->sevents
&= ~POLLMSG
;
788 wakeup(&gref
->event
);
791 if (gref
->sevents
& POLLIN
) {
792 gref
->sevents
&= ~POLLIN
;
793 selwakeup(&gref
->si
);
799 void atalk_notify_sel(gref
)
803 if (gref
->sevents
& POLLIN
) {
804 gref
->sevents
&= ~POLLIN
;
805 selwakeup(&gref
->si
);
809 int atalk_peek(gref
, event
)
811 unsigned char *event
;
816 *event
= *gbuf_rptr(gref
->rdhead
);
825 static gbuf_t
*trace_msg
;
827 void atalk_settrace(char * str
, p1
, p2
, p3
, p4
, p5
)
833 sprintf(trace_buf
, str
, p1
, p2
, p3
, p4
, p5
);
834 len
= strlen(trace_buf
);
835 #ifdef APPLETALK_DEBUG
836 kprintf("atalk_settrace: gbufalloc size=%d\n", len
+1);
838 if ((m
= gbuf_alloc(len
+1, PRI_MED
)) == 0)
841 strcpy(gbuf_rptr(m
), trace_buf
);
843 for (nextm
=trace_msg
; gbuf_cont(nextm
); nextm
=gbuf_cont(nextm
)) ;
844 gbuf_cont(nextm
) = m
;
849 void atalk_gettrace(m
)
853 gbuf_cont(m
) = trace_msg
;
859 #define GREF_PER_BLK 32
860 static gref_t
*gref_free_list
= 0;
861 extern gbuf_t
*atp_resource_m
;
863 int gref_alloc(grefp
)
868 gref_t
*gref
, *gref_array
;
870 *grefp
= (gref_t
*)NULL
;
872 if (gref_free_list
== 0) {
873 #ifdef APPLETALK_DEBUG
874 kprintf("gref_alloc: gbufalloc size=%d\n", GREF_PER_BLK
*sizeof(gref_t
));
876 if ((m
= gbuf_alloc(GREF_PER_BLK
*sizeof(gref_t
),PRI_HI
)) == 0)
878 bzero(gbuf_rptr(m
), GREF_PER_BLK
*sizeof(gref_t
));
879 gref_array
= (gref_t
*)gbuf_rptr(m
);
880 for (i
=0; i
< GREF_PER_BLK
-1; i
++)
881 gref_array
[i
].atpcb_next
= (gref_t
*)&gref_array
[i
+1];
882 gbuf_cont(m
) = atp_resource_m
;
884 gref_array
[i
].atpcb_next
= gref_free_list
;
885 gref_free_list
= (gref_t
*)&gref_array
[0];
888 gref
= gref_free_list
;
889 gref_free_list
= gref
->atpcb_next
;
890 ATEVENTINIT(gref
->event
);
891 ATEVENTINIT(gref
->iocevent
);
893 /* *** just for now *** */
894 gref
->atpcb_socket
= (struct socket
*)NULL
;
900 /* bms: make gref_close callable from kernel */
901 int gref_close(gref_t
*gref
)
905 switch (gref
->proto
) {
907 /* ATPROTO_DDP and ATPROTO_LAP have been replaced with
908 BSD-style socket interface. */
911 rc
= atp_close(gref
, 1); break;
913 rc
= asp_close(gref
); break;
916 rc
= aurp_close(gref
); break;
920 rc
= adsp_close(gref
); break;
928 selthreadclear(&gref
->si
);
930 /* from original gref_free() */
931 bzero((char *)gref
, sizeof(gref_t
));
932 gref
->atpcb_next
= gref_free_list
;
933 gref_free_list
= gref
;
940 temp fix for bug 2731148 - until this code is re-written to use standard clusters
941 Deletes any free clusters on the free list.
943 void atp_delete_free_clusters(__unused
void *junk
)
946 caddr_t cluster_list
;
948 /* check for free clusters on the free_cluster_list to be deleted */
950 untimeout(&atp_delete_free_clusters
, NULL
);
952 lck_mtx_lock(atalk_cluster_lock
);
954 atp_free_cluster_timeout_set
= 0;
956 cluster_list
= atp_free_cluster_list
;
957 atp_free_cluster_list
= NULL
;
959 lck_mtx_unlock(atalk_cluster_lock
);
961 while ((cluster
= cluster_list
))
963 cluster_list
= *((caddr_t
*)cluster
);
964 FREE(cluster
, M_MCLUST
);
970 Used as the "free" routine for over-size clusters allocated using
974 void m_lgbuf_free(caddr_t
, u_int
, caddr_t
);
979 __unused caddr_t arg
) /* not needed, but they're in m_free() */
983 /* move to free_cluster_list to be deleted later */
984 caddr_t cluster
= (caddr_t
)buf
;
986 lck_mtx_lock(atalk_cluster_lock
);
988 *((caddr_t
*)cluster
) = atp_free_cluster_list
;
989 atp_free_cluster_list
= cluster
;
991 if ((t
= atp_free_cluster_timeout_set
) == 0)
992 atp_free_cluster_timeout_set
= 1;
994 lck_mtx_unlock(atalk_cluster_lock
);
997 timeout(&atp_delete_free_clusters
, NULL
, (1 * HZ
));
1001 Used to allocate an mbuf when there is the possibility that it may
1002 need to be larger than the size of a standard cluster.
1005 struct mbuf
*m_lgbuf_alloc(size
, wait
)
1010 if (atp_free_cluster_list
)
1011 atp_delete_free_clusters(NULL
); /* delete any free clusters on the free list */
1014 * check that the passed size is within admissible boundaries
1015 * The max data size being ASP of 4576 (8 * ATP_DATA_SIZE),
1016 * allow for extra space for control data
1019 if (size
< 0 || size
> (ATP_DATA_SIZE
* 10))
1022 /* If size is too large, allocate a cluster, otherwise, use the
1023 standard mbuf allocation routines.*/
1024 if (size
> MCLBYTES
) {
1027 (buf
= (void *)_MALLOC(size
, M_MCLUST
,
1028 (wait
)? M_WAITOK
: M_NOWAIT
))) {
1032 (m
= m_clattach(NULL
, MSG_DATA
, buf
, m_lgbuf_free
, size
, 0,
1033 (wait
)? M_WAIT
: M_DONTWAIT
))) {
1034 m_lgbuf_free(buf
, 0, 0);
1038 m
= m_gethdr(((wait
)? M_WAIT
: M_DONTWAIT
), MSG_DATA
);
1039 if (m
&& ((size_t)size
> MHLEN
)) {
1040 MCLGET(m
, ((wait
)? M_WAIT
: M_DONTWAIT
));
1041 if (!(m
->m_flags
& M_EXT
)) {
1049 } /* m_lgbuf_alloc */
1052 gbuf_alloc() is a wrapper for m_lgbuf_alloc(), which is used to
1053 allocate an mbuf when there is the possibility that it may need
1054 to be larger than the size of a standard cluster.
1056 gbuf_alloc() sets the mbuf lengths, unlike the standard mbuf routines.
1059 gbuf_t
*gbuf_alloc_wait(size
, wait
)
1062 gbuf_t
*m
= (gbuf_t
*)m_lgbuf_alloc(size
, wait
);
1064 /* Standard mbuf allocation routines assume that the caller
1065 will set the size. */
1067 m
->m_pkthdr
.len
= size
;
1079 for (size
=0; m
; m
=gbuf_cont(m
))
1080 size
+= gbuf_len(m
);
1084 int append_copy(m1
, m2
, wait
)
1085 struct mbuf
*m1
, *m2
;
1088 if ((!(m1
->m_flags
& M_EXT
)) && (!(m2
->m_flags
& M_EXT
)) &&
1089 (m_trailingspace(m1
) >= m2
->m_len
)) {
1090 /* splat the data from one into the other */
1091 bcopy(mtod(m2
, caddr_t
), mtod(m1
, caddr_t
) + m1
->m_len
,
1093 m1
->m_len
+= m2
->m_len
;
1094 if (m1
->m_flags
& M_PKTHDR
)
1095 m1
->m_pkthdr
.len
+= m2
->m_len
;
1098 if ((m1
->m_next
= m_copym(m2
, 0, m2
->m_len
,
1099 (wait
)? M_WAIT
: M_DONTWAIT
)) == NULL
)
1105 Copy an mbuf chain, referencing existing external storage, if any.
1106 Leave space for a header in the new chain, if the space has been
1107 left in the origin chain.
1109 struct mbuf
*copy_pkt(mlist
, pad
)
1110 struct mbuf
*mlist
; /* the mbuf chain to be copied */
1111 int pad
; /* hint as to how long the header might be
1112 If pad is < 0, leave the same amount of space
1113 as there was in the original. */
1119 len
= m_leadingspace(mlist
);
1121 len
= min(pad
, m_leadingspace(mlist
));
1123 /* preserve space for the header at the beginning of the mbuf */
1125 mlist
->m_data
-= (len
);
1126 mlist
->m_len
+= (len
);
1127 if (mlist
->m_flags
& M_PKTHDR
)
1128 mlist
->m_pkthdr
.len
+= (len
);
1129 new_m
= m_copym(mlist
, 0, M_COPYALL
, M_DONTWAIT
);
1133 new_m
= m_copym(mlist
, 0, M_COPYALL
, M_DONTWAIT
);
1138 void gbuf_linkb(m1
, m2
)
1142 while (gbuf_cont(m1
) != 0)
1147 void gbuf_linkpkt(m1
, m2
)
1151 while (gbuf_next(m1
) != 0)
1161 while ((tmp_m
= m
) != 0) {
1163 gbuf_next(tmp_m
) = 0;
1169 /* free empty mbufs at the front of the chain */
1170 gbuf_t
*gbuf_strip(m
)
1175 while (m
&& gbuf_len(m
) == 0) {
1183 /**************************************/
1185 int ddp_adjmsg(m
, len
)
1190 gbuf_t
*curr_m
, *prev_m
;
1192 if (m
== (gbuf_t
*)0)
1196 for (curr_m
=m
; curr_m
;) {
1197 buf_len
= gbuf_len(curr_m
);
1198 if (len
< buf_len
) {
1199 gbuf_rinc(curr_m
,len
);
1203 gbuf_rinc(curr_m
,buf_len
);
1204 if ((curr_m
= gbuf_cont(curr_m
)) == 0) {
1210 } else if (len
< 0) {
1213 for (curr_m
=m
; gbuf_cont(curr_m
);
1214 prev_m
=curr_m
, curr_m
=gbuf_cont(curr_m
)) ;
1215 buf_len
= gbuf_len(curr_m
);
1216 if (len
< buf_len
) {
1217 gbuf_wdec(curr_m
,len
);
1222 gbuf_cont(prev_m
) = 0;
1233 * The message chain, m is grown in size by len contiguous bytes.
1234 * If len is non-negative, len bytes are added to the
1235 * end of the gbuf_t chain. If len is negative, the
1236 * bytes are added to the front. ddp_growmsg only adds bytes to
1237 * message blocks of the same type.
1238 * It returns a pointer to the new gbuf_t on sucess, 0 on failure.
1241 gbuf_t
*ddp_growmsg(mp
, len
)
1247 if ((m
= mp
) == (gbuf_t
*) 0)
1248 return ((gbuf_t
*) 0);
1252 if ((d
= gbuf_alloc(len
, PRI_MED
)) == 0)
1253 return ((gbuf_t
*) 0);
1254 gbuf_set_type(d
, gbuf_type(m
));
1256 /* link in new gbuf_t */
1265 if ((count
= gbuf_msgsize(m
)) < 0)
1266 return ((gbuf_t
*) 0);
1267 /* find end of chain */
1268 for ( ; m
; m
= gbuf_cont(m
)) {
1269 if (gbuf_len(m
) >= count
)
1271 count
-= gbuf_len(m
);
1273 /* m now points to gbuf_t to add to */
1274 if ((d
= gbuf_alloc(len
, PRI_MED
)) == 0)
1275 return ((gbuf_t
*) 0);
1276 gbuf_set_type(d
, gbuf_type(m
));
1277 /* link in new gbuf_t */
1278 gbuf_cont(d
) = gbuf_cont(m
);
1286 * return the MSG_IOCACK/MSG_IOCNAK. Note that the same message
1287 * block is used as the vehicle, and that if there is an error return,
1288 * then linked blocks are lopped off. BEWARE of multiple references.
1289 * Used by other appletalk modules, so it is not static!
1292 void ioc_ack(errno
, m
, gref
)
1295 register gref_t
*gref
;
1297 ioc_t
*iocbp
= (ioc_t
*)gbuf_rptr(m
);
1299 /*kprintf("ioc_ack: m=%x gref=%x errno=%d\n", m, gref, errno);*/
1300 if ((iocbp
->ioc_error
= errno
) != 0)
1301 { /* errno != 0, then there is an error, get rid of linked blocks! */
1304 gbuf_freem(gbuf_cont(m
));
1307 gbuf_set_type(m
, MSG_IOCNAK
);
1308 iocbp
->ioc_count
= 0; /* only make zero length if error */
1309 iocbp
->ioc_rval
= -1;
1311 gbuf_set_type(m
, MSG_IOCACK
);
1313 atalk_putnext(gref
, m
);
1317 static void ioccmd_t_32_to_64( ioccmd_t
*from_p
, user_ioccmd_t
*to_p
)
1319 to_p
->ic_cmd
= from_p
->ic_cmd
;
1320 to_p
->ic_timout
= from_p
->ic_timout
;
1321 to_p
->ic_len
= from_p
->ic_len
;
1322 to_p
->ic_dp
= CAST_USER_ADDR_T(from_p
->ic_dp
);
1326 static void ioccmd_t_64_to_32( user_ioccmd_t
*from_p
, ioccmd_t
*to_p
)
1328 to_p
->ic_cmd
= from_p
->ic_cmd
;
1329 to_p
->ic_timout
= from_p
->ic_timout
;
1330 to_p
->ic_len
= from_p
->ic_len
;
1331 to_p
->ic_dp
= CAST_DOWN(caddr_t
, from_p
->ic_dp
);