Source to bsd/netat/adsp_attention.c
/*
* Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
*
* @APPLE_LICENSE_HEADER_START@
*
* The contents of this file constitute Original Code as defined in and
* are subject to the Apple Public Source License Version 1.1 (the
* "License"). You may not use this file except in compliance with the
* License. Please obtain a copy of the License at
* http://www.apple.com/publicsource and read it before using this file.
*
* This Original Code and all software distributed under the License are
* distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
* EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
* INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT. Please see the
* License for the specific language governing rights and limitations
* under the License.
*
* @APPLE_LICENSE_HEADER_END@
*/
/*
* dspAttention.c
*
* From Mike Shoemaker v01.05 03/16/90 mbs
*/
/*
* Change log:
* 06/29/95 - Modified to handle flow control for writing (Tuyen Nguyen)
* Modified for MP, 1996 by Tuyen Nguyen
* Modified, April 9, 1997 by Tuyen Nguyen for MacOSX.
*/
#include <sys/errno.h>
#include <sys/types.h>
#include <sys/param.h>
#include <machine/spl.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/proc.h>
#include <sys/filedesc.h>
#include <sys/fcntl.h>
#include <sys/mbuf.h>
#include <sys/socket.h>
#include <netat/sysglue.h>
#include <netat/appletalk.h>
#include <netat/at_pcb.h>
#include <netat/debug.h>
#include <netat/adsp.h>
#include <netat/adsp_internal.h>
/*
* dspAttention
*
* INPUTS:
* --> ccbRefNum refnum of connection end
* --> attnCode client attention code
* --> attnSize size in bytes of attention data
* --> attnData pointer to attention data
* --> attnInterval attention retransmit interval
* (ignored by ADSP 1.5 & up)
*
* OUTPUTS:
* none
*
* ERRORS:
* errRefNum bad connection refnum
* errState connection is not open
* errAttention attention message too long
* errAborted request aborted by Remove or Close call
*/
int adspAttention(sp, pb) /* (DSPPBPtr pb) */
register struct adspcmd *pb;
register CCBPtr sp;
{
int s;
register gbuf_t *mp, *nmp;
unsigned char uerr;
if (sp == 0) {
pb->ioResult = errRefNum;
return EINVAL;
}
if (sp->state != sOpen) { /* If we're not open, tell user to go away */
pb->ioResult = errState;
uerr = ENOTCONN;
l_err:
atalk_notify(sp->gref, uerr);
gbuf_freem(pb->mp);
return 0;
}
if (pb->u.attnParams.attnSize > attnBufSize) /* If data too big, bye-bye */
{
pb->ioResult = errAttention;
uerr = ERANGE;
goto l_err;
}
/* The 1st mbuf in the pb->mp chain (mp) is the adspcmd structure.
The 2nd mbuf (nmp) will be the beginning of the data. */
mp = pb->mp;
if (pb->u.attnParams.attnSize) {
nmp = gbuf_cont(mp);
if (gbuf_len(mp) > sizeof(struct adspcmd)) {
if ((nmp = gbuf_dupb(mp)) == 0) {
gbuf_wset(mp, sizeof(struct adspcmd));
uerr = ENOBUFS;
goto l_err;
}
gbuf_wset(mp, sizeof(struct adspcmd));
gbuf_rinc(nmp, sizeof(struct adspcmd));
gbuf_cont(nmp) = gbuf_cont(mp);
gbuf_cont(mp) = nmp;
}
}
pb->ioDirection = 1; /* outgoing attention data */
ATDISABLE(s, sp->lock);
if (sp->sapb) { /* Pending attentions already? */
qAddToEnd(&sp->sapb, pb); /* Just add to end of queue */
ATENABLE(s, sp->lock);
} else {
sp->sendAttnData = 1; /* Start off this attention */
pb->qLink = 0;
sp->sapb = pb;
ATENABLE(s, sp->lock);
CheckSend(sp);
}
pb->ioResult = 1; /* indicate that the IO is not complete */
return 0;
}