ref: 17f7f6be4e1a316c0f5f26ff70e047aece4de2bc
dir: /sys/src/cmd/cifs/pack.c/
/* packet packing and unpacking */ #include <u.h> #include <libc.h> #include <ctype.h> #include "cifs.h" void * pmem(Pkt *p, void *v, int len) { uchar *str = v; void *s = p->pos; if(!len || !v) return s; while(len--) *p->pos++ = *str++; return s; } void * ppath(Pkt *p, char *str) { char c; Rune r; void *s = p->pos; if(!str) return s; if(p->s->flags2 & FL2_UNICODE){ if(((p->pos - p->buf) % 2) != 0) /* pad to even offset */ p8(p, 0); while(*str){ str += chartorune(&r, str); if(r > Bits16) sysfatal("ppath: %C/%x utf too wide for windows\n", r, r); if(r == L'/') r = L'\\'; pl16(p, r); } pl16(p, 0); } else { while((c = *str++) != 0){ if(c == '/') c = '\\'; *p->pos++ = c; } *p->pos++ = 0; } return s; } void * pstr(Pkt *p, char *str) { void *s = p->pos; Rune r; if(!str) return s; if(p->s->flags2 & FL2_UNICODE){ if(((p->pos - p->buf) % 2) != 0) p8(p, 0); /* pad to even offset */ while(*str){ str += chartorune(&r, str); if(r > Bits16) sysfatal("pstr: %C/%x utf too wide for windows\n", r, r); pl16(p, r); } pl16(p, 0); } else { while(*str) *p->pos++ = *str++; *p->pos++ = 0; } return s; } void * pascii(Pkt *p, char *str) { void *s = p->pos; while(*str) *p->pos++ = *str++; *p->pos++ = 0; return s; } void * pl64(Pkt *p, uvlong n) { void *s = p->pos; *p->pos++ = n; *p->pos++ = n >> 8; *p->pos++ = n >> 16; *p->pos++ = n >> 24; *p->pos++ = n >> 32; *p->pos++ = n >> 40; *p->pos++ = n >> 48; *p->pos++ = n >> 56; return s; } void * pb32(Pkt *p, uint n) { void *s = p->pos; *p->pos++ = n >> 24; *p->pos++ = n >> 16; *p->pos++ = n >> 8; *p->pos++ = n; return s; } void * pl32(Pkt *p, uint n) { void *s = p->pos; *p->pos++ = n; *p->pos++ = n >> 8; *p->pos++ = n >> 16; *p->pos++ = n >> 24; return s; } void * pb16(Pkt *p, uint n) { void *s = p->pos; *p->pos++ = n >> 8; *p->pos++ = n; return s; } void * pl16(Pkt *p, uint n) { void *s = p->pos; *p->pos++ = n; *p->pos++ = n >> 8; return s; } void * p8(Pkt *p, uint n) { void *s = p->pos; *p->pos++ = n; return s; } /* * Encode a Netbios name */ void * pname(Pkt *p, char *name, char pad) { int i, done = 0; char c; void *s = p->pos; *p->pos++ = ' '; for(i = 0; i < 16; i++) { c = pad; if(!done && name[i] == '\0') done = 1; if(!done) c = islower(name[i])? toupper(name[i]): name[i]; *p->pos++ = ((uchar)c >> 4) + 'A'; *p->pos++ = (c & 0xf) + 'A'; } *p->pos++ = '\0'; return s; } void * pvtime(Pkt *p, uvlong n) { void *s = p->pos; n += 11644473600LL; n *= 10000000LL; pl32(p, n); pl32(p, n >> 32); return s; } void * pdatetime(Pkt *p, long utc) { void *s = p->pos; Tm *tm = localtime(utc); int t = tm->hour << 11 | tm->min << 5 | (tm->sec / 2); int d = (tm->year - 80) << 9 | (tm->mon + 1) << 5 | tm->mday; /* * bug in word swapping in Win95 requires this */ if(p->s->caps & CAP_NT_SMBS){ pl16(p, d); pl16(p, t); } else{ pl16(p, t); pl16(p, d); } return s; } void gmem(Pkt *p, void *v, int n) { uchar *str = v; if(!n || !v) return; while(n-- && p->pos < p->eop) *str++ = *p->pos++; } /* * note len is the length of the source string in * in runes or bytes, in ASCII mode this is also the size * of the output buffer but this is not so in Unicode mode! */ static void _gstr(Pkt *p, char *str, int n, int align) { int i; Rune r; if(!n || !str) return; if(p->flags2 & FL2_UNICODE){ if(((p->pos - p->buf) % 2) != 0){ if(align) abort(); } i = 0; while(*p->pos && n && p->pos < p->eop){ r = gl16(p); i += runetochar(str +i, &r); n -= 2; } *(str + i) = 0; while(*p->pos && p->pos < p->eop) gl16(p); /* * some versions of windows terminate a rune string * with a single nul so we do a dangerous hack... */ if(p->pos[1]) g8(p); else gl16(p); } else { while(*p->pos && n-- && p->pos < p->eop) *str++ = *p->pos++; *str = 0; while(*p->pos++ && p->pos < p->eop) continue; } } void gstr(Pkt *p, char *str, int n) { _gstr(p, str, n, 1); } void gstr_noalign(Pkt *p, char *str, int n) { _gstr(p, str, n, 0); } /* * Because DFS uses a string heap rather than strings embedded in the * data packet, experience (rather than any spec) tells, us we must * turn off the 16bit alignment for unicode strings. */ void goff(Pkt *p, uchar *base, char *str, int n) { int off; uchar *pos; off = gl16(p); if(off == 0 || base + off > p->eop){ memset(str, 0, n); return; } pos = p->pos; p->pos = base + off; gstr_noalign(p, str, n); p->pos = pos; } void gascii(Pkt *p, char *str, int n) { if(!n || !str) return; while(*p->pos && n-- && p->pos < p->eop) *str++ = *p->pos++; *str = 0; while(*p->pos++ && p->pos < p->eop) continue; } uvlong gl64(Pkt *p) { uvlong n; if(p->pos + 8 > p->eop) return 0; n = (uvlong)*p->pos++; n |= (uvlong)*p->pos++ << 8; n |= (uvlong)*p->pos++ << 16; n |= (uvlong)*p->pos++ << 24; n |= (uvlong)*p->pos++ << 32; n |= (uvlong)*p->pos++ << 40; n |= (uvlong)*p->pos++ << 48; n |= (uvlong)*p->pos++ << 56; return n; } uvlong gb48(Pkt *p) { uvlong n; if(p->pos + 6 > p->eop) return 0; n = (uvlong)*p->pos++ << 40; n |= (uvlong)*p->pos++ << 24; n |= (uvlong)*p->pos++ << 32; n |= (uvlong)*p->pos++ << 16; n |= (uvlong)*p->pos++ << 8; n |= (uvlong)*p->pos++; return n; } uint gb32(Pkt *p) { uint n; if(p->pos + 4 > p->eop) return 0; n = (uint)*p->pos++ << 24; n |= (uint)*p->pos++ << 16; n |= (uint)*p->pos++ << 8; n |= (uint)*p->pos++; return n; } uint gl32(Pkt *p) { uint n; if(p->pos + 4 > p->eop) return 0; n = (uint)*p->pos++; n |= (uint)*p->pos++ << 8; n |= (uint)*p->pos++ << 16; n |= (uint)*p->pos++ << 24; return n; } uint gb16(Pkt *p) { uint n; if(p->pos + 2 > p->eop) return 0; n = (uint)*p->pos++ << 8; n |= (uint)*p->pos++; return n; } uint gl16(Pkt *p) { uint n; if(p->pos + 2 > p->eop) return 0; n = (uint)*p->pos++; n |= (uint)*p->pos++ << 8; return n; } uint g8(Pkt *p) { if(p->pos + 1 > p->eop) return 0; return (uint)*p->pos++; } long gdatetime(Pkt *p) { Tm tm; uint d, t; if(p->pos + 4 > p->eop) return 0; /* * bug in word swapping in Win95 requires this */ if(p->s->caps & CAP_NT_SMBS){ d = gl16(p); t = gl16(p); }else{ t = gl16(p); d = gl16(p); } memset(&tm, 0, sizeof(tm)); tm.year = 80 + (d >> 9); tm.mon = ((d >> 5) & 017) - 1; tm.mday = d & 037; tm.zone[0] = 0; tm.tzoff = p->s->tz; tm.hour = t >> 11; tm.min = (t >> 5) & 63; tm.sec = (t & 31) << 1; return tm2sec(&tm); } long gvtime(Pkt *p) { uvlong vl; if(p->pos + 8 > p->eop) return 0; vl = (uvlong)gl32(p); vl |= (uvlong)gl32(p) << 32; vl /= 10000000LL; vl -= 11644473600LL; return vl; } void gconv(Pkt *p, int conv, char *str, int n) { int off; uchar *pos; off = gl32(p) & 0xffff; if(off == 0 || p->tdata - conv + off > p->eop){ memset(str, 0, n); return; } pos = p->pos; p->pos = p->tdata - conv + off; gascii(p, str, n); p->pos = pos; }