#include "textcode.h" #include "haveinline.h" static inline int dec(unsigned char x) { if (x>='A' && x<='Z') return x-'A'; if (x>='a' && x<='z') return x-'a'+26; if (x>='0' && x<='9') return x-'0'+26+26; switch (x) { case '-': return 62; case '_': return 63; default: return -1; } } size_t scan_base64url(const char *src,char *dest,size_t *destlen) { unsigned short tmp=0,bits=0; register const unsigned char* s=(const unsigned char*) src; size_t i; for (i=0;;) { int a=dec(*s); if (a<0) break; /* base64url does not have padding */ tmp=(tmp<<6)|a; bits+=6; ++s; if (bits>=8) { bits-=8; if (dest) dest[i]=(tmp>>bits); ++i; } } if (destlen) *destlen=i; return (const char*)s-src; } #ifdef UNITTEST #include <assert.h> #include <string.h> #include <stdio.h> int main() { char buf[100]; size_t i,l; /* check that we don't consume padding */ memset(buf,0,10); assert(scan_base64url("Zm5vcmQ=",buf,&l)==7 && l==5 && !memcmp(buf,"fnord",6)); /* check that we don't insist on the padding */ memset(buf,0,10); assert(scan_base64url("Zm5vcmQ",buf,&l)==7 && l==5 && !memcmp(buf,"fnord",6)); /* check the special non-isalnum chars :) */ memset(buf,0,10); assert(scan_base64url("_-8=",buf,&l)==3 && l==2 && !memcmp(buf,"\xff\xef",3)); return 0; } #endif