fastfetch-sys 2.43.0

A neofetch like system information tool
Documentation
#include "base64.h"

// https://github.com/kostya/benchmarks/blob/master/base64/test-nolib.c#L145
void ffBase64EncodeRaw(uint32_t size, const char *str, uint32_t *out_size, char *output)
{
    static const char chars[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
    char *out = output;
    const char *ends = str + (size - size % 3);
    while (str != ends)
    {
        uint32_t n = __builtin_bswap32(*(uint32_t *)str);
        *out++ = chars[(n >> 26) & 63];
        *out++ = chars[(n >> 20) & 63];
        *out++ = chars[(n >> 14) & 63];
        *out++ = chars[(n >> 8) & 63];
        str += 3;
    }

    if (size % 3 == 1)
    {
        uint64_t n = (uint64_t)*str << 16;
        *out++ = chars[(n >> 18) & 63];
        *out++ = chars[(n >> 12) & 63];
        *out++ = '=';
        *out++ = '=';
    }
    else if (size % 3 == 2)
    {
        uint64_t n = (uint64_t)*str++ << 16;
        n |= (uint64_t)*str << 8;
        *out++ = chars[(n >> 18) & 63];
        *out++ = chars[(n >> 12) & 63];
        *out++ = chars[(n >> 6) & 63];
        *out++ = '=';
    }
    *out = '\0';
    *out_size = (uint32_t)(out - output);
}

static uint8_t decode_table[256];

void init_decode_table()
{
    uint8_t ch = 0;
    do
    {
        int32_t code = -1;
        if (ch >= 'A' && ch <= 'Z')
            code = ch - 0x41;
        if (ch >= 'a' && ch <= 'z')
            code = ch - 0x47;
        if (ch >= '0' && ch <= '9')
            code = ch + 0x04;
        if (ch == '+' || ch == '-')
            code = 0x3E;
        if (ch == '/' || ch == '_')
            code = 0x3F;
        decode_table[ch] = (uint8_t) code;
    } while (ch++ < 0xFF);
}

#define next_char(x) uint8_t x = decode_table[(uint8_t) *str++];

bool ffBase64DecodeRaw(uint32_t size, const char *str, uint32_t *out_size, char *output)
{
    if (*(uint64_t*) decode_table == 0)
        init_decode_table();

    char *out = output;
    while (size > 0 && (str[size - 1] == '\n' || str[size - 1] == '\r' || str[size - 1] == '='))
        size--;

    const char *ends = str + size - 4;
    while (true)
    {
        if (str > ends)
            break;
        while (*str == '\n' || *str == '\r')
            str++;

        if (str > ends)
            break;
        next_char(a);
        next_char(b);
        next_char(c);
        next_char(d);

        *out++ = (char)(a << 2 | b >> 4);
        *out++ = (char)(b << 4 | c >> 2);
        *out++ = (char)(c << 6 | d >> 0);
    }

    uint8_t mod = (uint8_t) (ends - str + 4) % 4;
    if (mod == 2)
    {
        next_char(a);
        next_char(b);
        *out++ = (char)(a << 2 | b >> 4);
    }
    else if (mod == 3)
    {
        next_char(a);
        next_char(b);
        next_char(c);
        *out++ = (char)(a << 2 | b >> 4);
        *out++ = (char)(b << 4 | c >> 2);
    }

    *out = '\0';
    *out_size = (uint32_t) (out - output);
    return true;
}