ref: 04b419c602ed75deaed50b89fa164660fd679e8f
dir: /libsec/port/hmac.c/
#include "os.h" #include "libsec.h" /* rfc2104 */ DigestState* hmac_x(uchar *p, u32 len, uchar *key, u32 klen, uchar *digest, DigestState *s, DigestState*(*x)(uchar*, u32, uchar*, DigestState*), int xlen) { int i; uchar pad[Hmacblksz+1], innerdigest[256]; if(xlen > sizeof(innerdigest)) return nil; if(klen > Hmacblksz){ if(xlen > Hmacblksz) return nil; (*x)(key, klen, innerdigest, nil); key = innerdigest; klen = xlen; } /* first time through */ if(s == nil || s->seeded == 0){ memset(pad, 0x36, Hmacblksz); pad[Hmacblksz] = 0; for(i = 0; i < klen; i++) pad[i] ^= key[i]; s = (*x)(pad, Hmacblksz, nil, s); if(s == nil) return nil; } s = (*x)(p, len, nil, s); if(digest == nil) return s; /* last time through */ memset(pad, 0x5c, Hmacblksz); pad[Hmacblksz] = 0; for(i = 0; i < klen; i++) pad[i] ^= key[i]; (*x)(nil, 0, innerdigest, s); s = (*x)(pad, Hmacblksz, nil, nil); (*x)(innerdigest, xlen, digest, s); return nil; }