/* BLURB lgpl

                           Coda File System
                              Release 6

          Copyright (c) 1987-2003 Carnegie Mellon University
                  Additional copyrights listed below

This  code  is  distributed "AS IS" without warranty of any kind under
the  terms of the  GNU  Library General Public Licence  Version 2,  as
shown in the file LICENSE. The technical and financial contributors to
Coda are listed in the file CREDITS.

                        Additional copyrights
                           none currently

#*/

#include <stdlib.h>
#include "base64.h"

/* base 64 encoding/decoding to store the tokens in a convenient fileformat */
static char *b2e =
    "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";

void coda_base64_encode(FILE *out, char *in, int len)
{
    int i;

    for (i = 0; i < len; i += 3) {
        if ((i % 54) == 0) fputc('\n', out);

        fputc(b2e[(in[i] >> 2) & 0x3f], out);

        if (i+1 < len)
            fputc(b2e[((in[i] << 4) & 0x30) | ((in[i+1] >> 4) & 0xf)], out);
        else {
            fputc(b2e[((in[i] << 4) & 0x30)], out);
            fputc('=', out);
            fputc('=', out);
            break;
        }

        if (i+2 < len) {
            fputc(b2e[((in[i+1] << 2) & 0x3c) | ((in[i+2] >> 6) & 0x3)], out);
            fputc(b2e[in[i+2] & 0x3f], out);
        } else {
            fputc(b2e[((in[i+1] << 2) & 0x3c)], out);
            fputc('=', out);
        }
    }
    fputc('\n', out);
}

void coda_base64_decode(FILE *in, char **out, int *len)
{
    int val = 0, s = 18, n = 0, c, done = 0;
        
    *len = 24; *out = malloc(*len);

    while((c = fgetc(in)) != EOF) {
        if (c == '\n' || c == '\r') continue;
        if (c != '=') {
            if      (c >= 'A' && c <= 'Z') c =  c - 'A';
            else if (c >= 'a' && c <= 'z') c = (c - 'a') + 26;
            else if (c >= '0' && c <= '9') c = (c - '0') + 52;
            else if (c == '+')             c = 62;
            else if (c == '/')             c = 63;
	    else continue;
            val = val | (c << s);
        } else
            done = (s / 6) + 1;

        if ((s -= 6) < 0 || done) {
            (*out)[n]   = (val >> 16) & 0xff;
            (*out)[n+1] = (val >> 8) & 0xff;
            (*out)[n+2] = val & 0xff;
            val = 0; s = 18;
            n += 3 - done;
            if (done) break;

            if (n == *len) {
                *len += 24; *out = realloc(*out, *len);
            }
        }
    }
    *len = n;
}

#ifdef TESTING
#include <sys/stat.h>
#include <stdio.h>

#define TESTFILE "/tmp/base64_test"

void main(int argc, char **argv)
{
    FILE *fp;
    struct stat sb;
    char *outbuf = "testing the encoding", *inbuf;
    int   outlen, inlen;

    if (argc > 1)
        outbuf = argv[1];
    outlen = strlen(outbuf);

    if (stat(TESTFILE, &sb) == 0) {
        unlink(TESTFILE);
    }

    fp = fopen(TESTFILE, "w");
    coda_base64_encode(fp, outbuf, outlen);
    fclose(fp);
    
    fp = fopen(TESTFILE, "r");
    coda_base64_decode(fp, &inbuf, &inlen);
    fclose(fp);

    if (outlen != inlen || memcmp(outbuf, inbuf, outlen) != 0)
        puts("Error encoding/decoding: ");

    puts(inbuf);
    putchar('\n');
}
#endif



syntax highlighted by Code2HTML, v. 0.9.1