📄 rpc.c
字号:
#include <u.h>#include <libc.h>#include <thread.h>#include <sunrpc.h>/* * RPC protocol constants */enum{ RpcVersion = 2, /* msg type */ MsgCall = 0, MsgReply = 1, /* reply stat */ MsgAccepted = 0, MsgDenied = 1, /* accept stat */ MsgSuccess = 0, MsgProgUnavail = 1, MsgProgMismatch = 2, MsgProcUnavail = 3, MsgGarbageArgs = 4, MsgSystemErr = 5, /* reject stat */ MsgRpcMismatch = 0, MsgAuthError = 1, /* msg auth xxx */ MsgAuthOk = 0, MsgAuthBadCred = 1, MsgAuthRejectedCred = 2, MsgAuthBadVerf = 3, MsgAuthRejectedVerf = 4, MsgAuthTooWeak = 5, MsgAuthInvalidResp = 6, MsgAuthFailed = 7,};SunStatussunRpcPack(uchar *a, uchar *ea, uchar **pa, SunRpc *rpc){ u32int x; if(sunUint32Pack(a, ea, &a, &rpc->xid) < 0) goto Err; if(rpc->iscall){ if(sunUint32Pack(a, ea, &a, (x=MsgCall, &x)) < 0 || sunUint32Pack(a, ea, &a, (x=RpcVersion, &x)) < 0 || sunUint32Pack(a, ea, &a, &rpc->prog) < 0 || sunUint32Pack(a, ea, &a, &rpc->vers) < 0 || sunUint32Pack(a, ea, &a, &rpc->proc) < 0 || sunAuthInfoPack(a, ea, &a, &rpc->cred) < 0 || sunAuthInfoPack(a, ea, &a, &rpc->verf) < 0 || sunFixedOpaquePack(a, ea, &a, rpc->data, rpc->ndata) < 0) goto Err; }else{ if(sunUint32Pack(a, ea, &a, (x=MsgReply, &x)) < 0) goto Err; switch(rpc->status&0xF0000){ case 0: case SunAcceptError: if(sunUint32Pack(a, ea, &a, (x=MsgAccepted, &x)) < 0 || sunAuthInfoPack(a, ea, &a, &rpc->verf) < 0) goto Err; break; default: if(sunUint32Pack(a, ea, &a, (x=MsgDenied, &x)) < 0) goto Err; break; } switch(rpc->status){ case SunSuccess: if(sunUint32Pack(a, ea, &a, (x=MsgSuccess, &x)) < 0 || sunFixedOpaquePack(a, ea, &a, rpc->data, rpc->ndata) < 0) goto Err; break; case SunRpcMismatch: case SunProgMismatch: if(sunUint32Pack(a, ea, &a, (x=rpc->status&0xFFFF, &x)) < 0 || sunUint32Pack(a, ea, &a, &rpc->low) < 0 || sunUint32Pack(a, ea, &a, &rpc->high) < 0) goto Err; break; default: if(sunUint32Pack(a, ea, &a, (x=rpc->status&0xFFFF, &x)) < 0) goto Err; break; } } *pa = a; return SunSuccess;Err: *pa = ea; return SunGarbageArgs;}uintsunRpcSize(SunRpc *rpc){ uint a; a = 4; if(rpc->iscall){ a += 5*4; a += sunAuthInfoSize(&rpc->cred); a += sunAuthInfoSize(&rpc->verf); a += sunFixedOpaqueSize(rpc->ndata); }else{ a += 4; switch(rpc->status&0xF0000){ case 0: case SunAcceptError: a += 4+sunAuthInfoSize(&rpc->verf); break; default: a += 4; break; } switch(rpc->status){ case SunSuccess: a += 4+sunFixedOpaqueSize(rpc->ndata); break; case SunRpcMismatch: case SunProgMismatch: a += 3*4; default: a += 4; } } return a;}SunStatussunRpcUnpack(uchar *a, uchar *ea, uchar **pa, SunRpc *rpc){ u32int x; memset(rpc, 0, sizeof *rpc); if(sunUint32Unpack(a, ea, &a, &rpc->xid) < 0 || sunUint32Unpack(a, ea, &a, &x) < 0) goto Err; switch(x){ default: goto Err; case MsgCall: rpc->iscall = 1; if(sunUint32Unpack(a, ea, &a, &x) < 0 || x != RpcVersion || sunUint32Unpack(a, ea, &a, &rpc->prog) < 0 || sunUint32Unpack(a, ea, &a, &rpc->vers) < 0 || sunUint32Unpack(a, ea, &a, &rpc->proc) < 0 || sunAuthInfoUnpack(a, ea, &a, &rpc->cred) < 0 || sunAuthInfoUnpack(a, ea, &a, &rpc->verf) < 0) goto Err; rpc->ndata = ea-a; rpc->data = a; a = ea; break; case MsgReply: rpc->iscall = 0; if(sunUint32Unpack(a, ea, &a, &x) < 0) goto Err; switch(x){ default: goto Err; case MsgAccepted: if(sunAuthInfoUnpack(a, ea, &a, &rpc->verf) < 0 || sunUint32Unpack(a, ea, &a, &x) < 0) goto Err; switch(x){ case MsgSuccess: rpc->status = SunSuccess; rpc->ndata = ea-a; rpc->data = a; a = ea; break; case MsgProgUnavail: case MsgProcUnavail: case MsgGarbageArgs: case MsgSystemErr: rpc->status = SunAcceptError | x; break; case MsgProgMismatch: rpc->status = SunAcceptError | x; if(sunUint32Unpack(a, ea, &a, &rpc->low) < 0 || sunUint32Unpack(a, ea, &a, &rpc->high) < 0) goto Err; break; } break; case MsgDenied: if(sunUint32Unpack(a, ea, &a, &x) < 0) goto Err; switch(x){ default: goto Err; case MsgAuthError: if(sunUint32Unpack(a, ea, &a, &x) < 0) goto Err; rpc->status = SunAuthError | x; break; case MsgRpcMismatch: rpc->status = SunRejectError | x; if(sunUint32Unpack(a, ea, &a, &rpc->low) < 0 || sunUint32Unpack(a, ea, &a, &rpc->high) < 0) goto Err; break; } break; } } *pa = a; return SunSuccess;Err: *pa = ea; return SunGarbageArgs;}voidsunRpcPrint(Fmt *fmt, SunRpc *rpc){ fmtprint(fmt, "xid=%#ux", rpc->xid); if(rpc->iscall){ fmtprint(fmt, " prog %#ux vers %#ux proc %#ux [", rpc->prog, rpc->vers, rpc->proc); sunAuthInfoPrint(fmt, &rpc->cred); fmtprint(fmt, "] ["); sunAuthInfoPrint(fmt, &rpc->verf); fmtprint(fmt, "]"); }else{ fmtprint(fmt, " status %#ux [", rpc->status); sunAuthInfoPrint(fmt, &rpc->verf); fmtprint(fmt, "] low %#ux high %#ux", rpc->low, rpc->high); }}voidsunAuthInfoPrint(Fmt *fmt, SunAuthInfo *ai){ switch(ai->flavor){ case SunAuthNone: fmtprint(fmt, "none"); break; case SunAuthShort: fmtprint(fmt, "short"); break; case SunAuthSys: fmtprint(fmt, "sys"); break; default: fmtprint(fmt, "%#ux", ai->flavor); break; }// if(ai->ndata)// fmtprint(fmt, " %.*H", ai->ndata, ai->data);}uintsunAuthInfoSize(SunAuthInfo *ai){ return 4 + sunVarOpaqueSize(ai->ndata);}intsunAuthInfoPack(uchar *a, uchar *ea, uchar **pa, SunAuthInfo *ai){ if(sunUint32Pack(a, ea, &a, &ai->flavor) < 0 || sunVarOpaquePack(a, ea, &a, &ai->data, &ai->ndata, 400) < 0) goto Err; *pa = a; return 0;Err: *pa = ea; return -1;}intsunAuthInfoUnpack(uchar *a, uchar *ea, uchar **pa, SunAuthInfo *ai){ if(sunUint32Unpack(a, ea, &a, &ai->flavor) < 0 || sunVarOpaqueUnpack(a, ea, &a, &ai->data, &ai->ndata, 400) < 0) goto Err; *pa = a; return 0;Err: *pa = ea; return -1;}intsunEnumPack(uchar *a, uchar *ea, uchar **pa, int *e){ u32int x; x = *e; return sunUint32Pack(a, ea, pa, &x);}intsunUint1Pack(uchar *a, uchar *ea, uchar **pa, u1int *u){ u32int x; x = *u; return sunUint32Pack(a, ea, pa, &x);}intsunUint32Pack(uchar *a, uchar *ea, uchar **pa, u32int *u){ u32int x; if(ea-a < 4) goto Err; x = *u; *a++ = x>>24; *a++ = x>>16; *a++ = x>>8; *a++ = x; *pa = a; return 0;Err: *pa = ea; return -1;}intsunEnumUnpack(uchar *a, uchar *ea, uchar **pa, int *e){ u32int x; if(sunUint32Unpack(a, ea, pa, &x) < 0) return -1; *e = x; return 0;}intsunUint1Unpack(uchar *a, uchar *ea, uchar **pa, u1int *u){ u32int x; if(sunUint32Unpack(a, ea, pa, &x) < 0 || (x!=0 && x!=1)){ *pa = ea; return -1; } *u = x; return 0;}intsunUint32Unpack(uchar *a, uchar *ea, uchar **pa, u32int *u){ u32int x; if(ea-a < 4) goto Err; x = *a++ << 24; x |= *a++ << 16; x |= *a++ << 8; x |= *a++; *pa = a; *u = x; return 0;Err: *pa = ea; return -1;}intsunUint64Unpack(uchar *a, uchar *ea, uchar **pa, u64int *u){ u32int x, y; if(sunUint32Unpack(a, ea, &a, &x) < 0 || sunUint32Unpack(a, ea, &a, &y) < 0) goto Err; *u = ((uvlong)x<<32) | y; *pa = a; return 0;Err: *pa = ea; return -1;}intsunUint64Pack(uchar *a, uchar *ea, uchar **pa, u64int *u){ u32int x, y; x = *u >> 32; y = *u; if(sunUint32Pack(a, ea, &a, &x) < 0 || sunUint32Pack(a, ea, &a, &y) < 0) goto Err; *pa = a; return 0;Err: *pa = ea; return -1;}uintsunStringSize(char *s){ return (4+strlen(s)+3) & ~3;}intsunStringUnpack(uchar *a, uchar *ea, uchar **pa, char **s, u32int max){ uchar *dat; u32int n; if(sunVarOpaqueUnpack(a, ea, pa, &dat, &n, max) < 0) goto Err; /* slide string down over length to make room for NUL */ memmove(dat-1, dat, n); dat[-1+n] = 0; *s = (char*)(dat-1); return 0;Err: return -1;}intsunStringPack(uchar *a, uchar *ea, uchar **pa, char **s, u32int max){ u32int n; n = strlen(*s); return sunVarOpaquePack(a, ea, pa, (uchar**)s, &n, max);}uintsunVarOpaqueSize(u32int n){ return (4+n+3) & ~3;}intsunVarOpaquePack(uchar *a, uchar *ea, uchar **pa, uchar **dat, u32int *ndat, u32int max){ if(*ndat > max || sunUint32Pack(a, ea, &a, ndat) < 0 || sunFixedOpaquePack(a, ea, &a, *dat, *ndat) < 0) goto Err; *pa = a; return 0;Err: *pa = ea; return -1;}intsunVarOpaqueUnpack(uchar *a, uchar *ea, uchar **pa, uchar **dat, u32int *ndat, u32int max){ if(sunUint32Unpack(a, ea, &a, ndat) < 0 || *ndat > max) goto Err; *dat = a; a += (*ndat+3)&~3; if(a > ea) goto Err; *pa = a; return 0;Err: *pa = ea; return -1;}uintsunFixedOpaqueSize(u32int n){ return (n+3) & ~3;}intsunFixedOpaquePack(uchar *a, uchar *ea, uchar **pa, uchar *dat, u32int n){ uint nn; nn = (n+3)&~3; if(a+nn > ea) goto Err; memmove(a, dat, n); if(nn > n) memset(a+n, 0, nn-n); a += nn; *pa = a; return 0;Err: *pa = ea; return -1;}intsunFixedOpaqueUnpack(uchar *a, uchar *ea, uchar **pa, uchar *dat, u32int n){ uint nn; nn = (n+3)&~3; if(a+nn > ea) goto Err; memmove(dat, a, n); a += nn; *pa = a; return 0;Err: *pa = ea; return -1;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -