Files
seta75D d6fe8fe829 Init
2021-10-11 22:19:34 -03:00

336 lines
6.7 KiB
C

static char sccsid[] = "@(#)18 1.10 src/bos/usr/ccs/lib/libc/xdr_mbuf.c, libcrpc, bos411, 9428A410j 10/25/93 20:43:48";
/*
* COMPONENT_NAME: LIBCRPC
*
* FUNCTIONS: xdrmbuf_destroy
* xdrmbuf_getbytes
* xdrmbuf_getlong
* xdrmbuf_getmbuf
* xdrmbuf_getpos
* xdrmbuf_init
* xdrmbuf_inline
* xdrmbuf_putbuf
* xdrmbuf_putbytes
* xdrmbuf_putlong
* xdrmbuf_setpos
*
*
* ORIGINS: 24,27
*
* This module contains IBM CONFIDENTIAL code. -- (IBM
* Confidential Restricted when combined with the aggregated
* modules for this product)
* SOURCE MATERIALS
*
* (C) COPYRIGHT International Business Machines Corp. 1989,1993
* All Rights Reserved
* US Government Users Restricted Rights - Use, duplication or
* disclosure restricted by GSA ADP Schedule Contract with IBM Corp.
*/
/*
#if defined(LIBC_SCCS) && !defined(lint)
static char sccsid[] = "@(#)xdr_mbuf.c 1.4 90/07/19 4.1NFSSRC Copyr 1990 Sun Micro";
#endif
*
* Copyright (c) 1988 by Sun Microsystems, Inc.
* 1.22 88/02/08
*/
/*
* xdr_mbuf.c, XDR implementation on kernel mbufs.
*/
#include <sys/types.h>
#include <sys/param.h>
#include <sys/mbuf.h>
#include <rpc/types.h>
#include <rpc/xdr.h>
#include <netinet/in.h>
#include <sys/uio.h>
#include <nfs/nfs_trc.h>
bool_t xdrmbuf_getlong(), xdrmbuf_putlong();
bool_t xdrmbuf_getbytes(), xdrmbuf_putbytes();
u_int xdrmbuf_getpos();
bool_t xdrmbuf_setpos();
long * xdrmbuf_inline();
void xdrmbuf_destroy();
/*
* Xdr on mbufs operations vector.
*/
struct xdr_ops xdrmbuf_ops = {
xdrmbuf_getlong,
xdrmbuf_putlong,
xdrmbuf_getbytes,
xdrmbuf_putbytes,
xdrmbuf_getpos,
xdrmbuf_setpos,
xdrmbuf_inline,
xdrmbuf_destroy
};
/*
* Initailize xdr stream.
*/
void
xdrmbuf_init(xdrs, m, op)
register XDR *xdrs;
register struct mbuf *m;
enum xdr_op op;
{
xdrs->x_op = op;
xdrs->x_ops = &xdrmbuf_ops;
xdrs->x_base = (caddr_t)m;
xdrs->x_private = mtod(m, caddr_t);
xdrs->x_public = (caddr_t)0;
xdrs->x_handy = m->m_len;
}
void
/* ARGSUSED */
xdrmbuf_destroy(xdrs)
XDR *xdrs;
{
/* do nothing */
}
bool_t
xdrmbuf_getlong(xdrs, lp)
register XDR *xdrs;
long *lp;
{
if ((xdrs->x_handy -= sizeof(long)) < 0) {
if (xdrs->x_handy != -sizeof(long))
TRCHKL0(HKWD_NFS_RPC_DEBUG|hkwd_KRPC_XDRMBUF_1);
if (xdrs->x_base) {
register struct mbuf *m =
((struct mbuf *)xdrs->x_base)->m_next;
while (m && m->m_len == 0) {
m = m->m_next;
}
xdrs->x_base = (caddr_t)m;
if (m == NULL)
return (FALSE);
xdrs->x_private = mtod(m, caddr_t);
xdrs->x_handy = m->m_len - sizeof(long);
} else {
return (FALSE);
}
}
*lp = ntohl(*((long *)(xdrs->x_private)));
xdrs->x_private += sizeof(long);
return (TRUE);
}
bool_t
xdrmbuf_putlong(xdrs, lp)
register XDR *xdrs;
long *lp;
{
if ((xdrs->x_handy -= sizeof(long)) < 0) {
if (xdrs->x_handy != -sizeof(long))
TRCHKL0(HKWD_NFS_RPC_DEBUG|hkwd_KRPC_XDRMBUF_2);
if (xdrs->x_base) {
register struct mbuf *m =
((struct mbuf *)xdrs->x_base)->m_next;
xdrs->x_base = (caddr_t)m;
if (m == NULL)
return (FALSE);
xdrs->x_private = mtod(m, caddr_t);
xdrs->x_handy = m->m_len - sizeof(long);
} else {
return (FALSE);
}
}
*(long *)xdrs->x_private = htonl(*lp);
xdrs->x_private += sizeof(long);
return (TRUE);
}
bool_t
xdrmbuf_getbytes(xdrs, addr, len)
register XDR *xdrs;
caddr_t addr;
register u_int len;
{
while ((xdrs->x_handy -= len) < 0) {
if ((xdrs->x_handy += len) > 0) {
bcopy(xdrs->x_private, addr, (u_int)xdrs->x_handy);
addr += xdrs->x_handy;
len -= xdrs->x_handy;
}
if (xdrs->x_base) {
register struct mbuf *m =
((struct mbuf *)xdrs->x_base)->m_next;
while (m && m->m_len == 0) {
m = m->m_next;
}
xdrs->x_base = (caddr_t)m;
if (m == NULL)
return (FALSE);
xdrs->x_private = mtod(m, caddr_t);
xdrs->x_handy = m->m_len;
} else {
return (FALSE);
}
}
bcopy(xdrs->x_private, addr, (u_int)len);
xdrs->x_private += len;
return (TRUE);
}
/*
* Sort of like getbytes except that instead of getting
* bytes we return the mbuf that contains all the rest
* of the bytes.
*/
bool_t
xdrmbuf_getmbuf(xdrs, mm, lenp)
register XDR *xdrs;
register struct mbuf **mm;
register u_int *lenp;
{
register struct mbuf *m;
register int len, used;
if (! xdr_u_int(xdrs, lenp)) {
return (FALSE);
}
m = (struct mbuf *)xdrs->x_base;
used = m->m_len - xdrs->x_handy;
m->m_data += used;
m->m_len -= used;
*mm = m;
/*
* Consistency check.
*/
len = 0;
while (m) {
len += m->m_len;
m = m->m_next;
}
if (len < *lenp) {
#ifdef _KERNEL
TRCHKL0(HKWD_NFS_RPC_DEBUG|hkwd_KRPC_XDRMBUF_3);
#endif
return (FALSE);
}
return (TRUE);
}
bool_t
xdrmbuf_putbytes(xdrs, addr, len)
register XDR *xdrs;
caddr_t addr;
register u_int len;
{
while ((xdrs->x_handy -= len) < 0) {
if ((xdrs->x_handy += len) > 0) {
bcopy(addr, xdrs->x_private, (u_int)xdrs->x_handy);
addr += xdrs->x_handy;
len -= xdrs->x_handy;
}
if (xdrs->x_base) {
register struct mbuf *m =
((struct mbuf *)xdrs->x_base)->m_next;
xdrs->x_base = (caddr_t)m;
if (m == NULL)
return (FALSE);
xdrs->x_private = mtod(m, caddr_t);
xdrs->x_handy = m->m_len;
} else {
return (FALSE);
}
}
bcopy(addr, xdrs->x_private, len);
xdrs->x_private += len;
return (TRUE);
}
/*
* Like putbytes, only we avoid the copy by pointing a external cluster
* mbuf at the buffer. Not safe if the buffer goes away before
* the mbuf chain is deallocated.
*/
bool_t
xdrmbuf_putbuf(xdrs, addr, len, func, arg)
register XDR *xdrs;
caddr_t addr;
u_int len;
int (*func)();
int arg;
{
register struct mbuf *m;
struct mbuf *m_clattach();
long llen = len;
if (len & 3) { /* can't handle roundup problems */
return (FALSE);
}
if (! xdrmbuf_putlong(xdrs, &llen)) {
return(FALSE);
}
((struct mbuf *)xdrs->x_base)->m_len -= xdrs->x_handy;
m = m_clattach(addr, func, (int)len, arg, M_WAIT);
if (m == NULL) {
TRCHKL0(HKWD_NFS_RPC_DEBUG|hkwd_KRPC_XDRMBUF_4);
return (FALSE);
}
((struct mbuf *)xdrs->x_base)->m_next = m;
xdrs->x_handy = 0;
return (TRUE);
}
u_int
xdrmbuf_getpos(xdrs)
register XDR *xdrs;
{
return (
(u_int)xdrs->x_private - mtod(((struct mbuf *)xdrs->x_base), u_int));
}
bool_t
xdrmbuf_setpos(xdrs, pos)
register XDR *xdrs;
u_int pos;
{
register caddr_t newaddr =
mtod(((struct mbuf *)xdrs->x_base), caddr_t) + pos;
register caddr_t lastaddr =
xdrs->x_private + xdrs->x_handy;
if ((int)newaddr > (int)lastaddr)
return (FALSE);
xdrs->x_private = newaddr;
xdrs->x_handy = (int)lastaddr - (int)newaddr;
return (TRUE);
}
long *
xdrmbuf_inline(xdrs, len)
register XDR *xdrs;
int len;
{
long *buf = 0;
if (xdrs->x_handy >= len) {
xdrs->x_handy -= len;
buf = (long *) xdrs->x_private;
xdrs->x_private += len;
}
return (buf);
}