#ifndef _NFS_NFSM_SUBS_H_
#define _NFS_NFSM_SUBS_H_
struct nfsm_info {
struct mbuf *nmi_mreq;
struct mbuf *nmi_mrep;
struct proc *nmi_procp;
struct ucred *nmi_cred;
struct mbuf *nmi_md;
struct mbuf *nmi_mb;
caddr_t nmi_dpos;
int nmi_v3;
int *nmi_errorp;
};
static inline void *
nfsm_dissect(struct nfsm_info *infop, int s)
{
caddr_t ret;
int avail, error;
avail = mtod(infop->nmi_md, caddr_t) + infop->nmi_md->m_len -
infop->nmi_dpos;
if (avail >= s) {
ret = infop->nmi_dpos;
infop->nmi_dpos += s;
return ret;
}
error = nfsm_disct(&infop->nmi_md, &infop->nmi_dpos, s, avail, &ret);
if (error != 0) {
m_freem(infop->nmi_mrep);
infop->nmi_mrep = NULL;
*infop->nmi_errorp = error;
return NULL;
} else {
return ret;
}
}
#define nfsm_rndup(a) (((a)+3)&(~0x3))
static inline int
nfsm_adv(struct nfsm_info *infop, int s)
{
int avail, error;
avail = mtod(infop->nmi_md, caddr_t) + infop->nmi_md->m_len -
infop->nmi_dpos;
if (avail >= s) {
infop->nmi_dpos += s;
return 0;
}
error = nfs_adv(&infop->nmi_md, &infop->nmi_dpos, s, avail);
if (error != 0) {
m_freem(infop->nmi_mrep);
infop->nmi_mrep = NULL;
*infop->nmi_errorp = error;
return error;
}
return 0;
}
static inline int
nfsm_postop_attr(struct nfsm_info *infop, struct vnode **vpp, int *attrflagp)
{
uint32_t *tl;
struct vnode *ttvp;
int attrflag, error;
if (infop->nmi_mrep == NULL)
return 0;
ttvp = *vpp;
tl = (uint32_t *)nfsm_dissect(infop, NFSX_UNSIGNED);
if (tl == NULL)
return 1;
attrflag = fxdr_unsigned(int, *tl);
if (attrflag != 0) {
error = nfs_loadattrcache(&ttvp, &infop->nmi_md,
&infop->nmi_dpos, NULL);
if (error != 0) {
m_freem(infop->nmi_mrep);
infop->nmi_mrep = NULL;
*infop->nmi_errorp = error;
return error;
}
*vpp = ttvp;
}
*attrflagp = attrflag;
return 0;
}
static inline int
nfsm_strsiz(struct nfsm_info *infop, int *lenp, int maxlen)
{
uint32_t *tl = (uint32_t *)nfsm_dissect(infop, NFSX_UNSIGNED);
int len;
if (tl == NULL)
return 1;
len = fxdr_unsigned(int32_t, *tl);
if (len < 0 || len > maxlen) {
m_freem(infop->nmi_mrep);
infop->nmi_mrep = NULL;
*infop->nmi_errorp = EBADRPC;
return 1;
}
*lenp = len;
return 0;
}
static inline int
nfsm_mtouio(struct nfsm_info *infop, struct uio *uiop, int len)
{
int error;
if (len <= 0)
return 0;
error = nfsm_mbuftouio(&infop->nmi_md, uiop, len, &infop->nmi_dpos);
if (error != 0) {
m_freem(infop->nmi_mrep);
infop->nmi_mrep = NULL;
*infop->nmi_errorp = error;
return error;
}
return 0;
}
static inline int
nfsm_strtom(struct nfsm_info *infop, char *str, size_t len, size_t maxlen)
{
if (len > maxlen) {
m_freem(infop->nmi_mreq);
infop->nmi_mreq = NULL;
*infop->nmi_errorp = ENAMETOOLONG;
return 1;
}
nfsm_strtombuf(&infop->nmi_mb, str, len);
return 0;
}
#endif