ppm: more methods, make existing one accessible
[libjh.git] / hex.c
1 // Copyright (2013) Jann Horn <jann@thejh.net>
2
3 HEADER #include <sys/types.h>
4 #include <assert.h>
5 #include <string.h>
6
7 static inline char hexchar(unsigned char c) {
8   if (c >= 10) return c+'a'-10;
9   return c+'0';
10 }
11
12 // If `in_len` is -1, it means that `in` is a null-terminated string.
13 // `out` must have space for `in_len*2+1` bytes.
14 PUBLIC_FN void hex_encode(char *in, ssize_t in_len, char *out) {
15   if (in_len == -1) in_len = strlen(in);
16   assert(in_len >= 0);
17   char *in_end = in+in_len;
18   while (in < in_end) {
19     unsigned char c = *(unsigned char *)(in++);
20     *(out++) = hexchar(c>>4);
21     *(out++) = hexchar(c&0xf);
22   }
23   *(out++) = '\0';
24 }
25
26 static inline unsigned char unhexchar(char c) {
27   if (c >= '0' && c <= '9') return c-'0';
28   if (c >= 'a' && c <= 'f') return c-'a'+10;
29   if (c >= 'A' && c <= 'F') return c-'A'+10;
30   return 255;
31 }
32
33 // Returns number of decoded bytes or -1(error).
34 // Does not null-terminate output.
35 PUBLIC_FN ssize_t hex_decode(char *in, ssize_t in_len, char *out) {
36   char *out_ = out;
37   if (in_len == -1) in_len = strlen(in);
38   assert(in_len >= 0);
39   if (in_len&1) return -1; /* n%2 must be 0 */
40   char *in_end = in + in_len;
41   while (in < in_end) {
42     unsigned char hi = unhexchar(*(in++));
43     unsigned char lo = unhexchar(*(in++));
44     if (hi == 255 || lo == 255) return -1;
45     *(out++) = (hi<<4) | lo;
46   }
47   return out_ - out;
48 }