cosmopolitan/tool/viz/lib/dither.c

124 lines
4.8 KiB
C

/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
╞══════════════════════════════════════════════════════════════════════════════╡
│ Copyright 2020 Justine Alexandra Roberts Tunney │
│ │
│ Permission to use, copy, modify, and/or distribute this software for │
│ any purpose with or without fee is hereby granted, provided that the │
│ above copyright notice and this permission notice appear in all copies. │
│ │
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
│ PERFORMANCE OF THIS SOFTWARE. │
╚─────────────────────────────────────────────────────────────────────────────*/
#include "dsp/tty/quant.h"
#include "libc/bits/hilbert.h"
#include "libc/bits/morton.h"
#include "libc/log/check.h"
#include "libc/log/log.h"
#include "libc/macros.h"
#include "libc/math.h"
#include "libc/mem/mem.h"
#include "libc/runtime/gc.h"
#include "libc/str/str.h"
#include "tool/viz/lib/graphic.h"
#include "tool/viz/lib/knobs.h"
#define CLAMP(X) MIN(255, MAX(0, X))
struct Dither {
struct {
int m5[32 * 32][2];
int m4[16 * 16][2];
int m3[8 * 8][2];
int m2[4 * 4][2];
int m1[2 * 2][2];
int m0[2 * 2][2];
} memory;
struct Chunk {
long b;
void *c;
} chunks[6];
} kDither;
static struct Chunk ChunkInit(long b, int c[b * b][2]) {
long i;
axdx_t h;
for (i = 0; i < b * b; ++i) {
h = unhilbert(b, i);
c[i][0] = h.ax;
c[i][1] = h.dx;
}
return (struct Chunk){b, c};
}
static optimizesize void DitherInit(struct Dither *d) {
d->chunks[0] = ChunkInit(1 << 0, d->memory.m0);
d->chunks[1] = ChunkInit(1 << 1, d->memory.m1);
d->chunks[2] = ChunkInit(1 << 2, d->memory.m2);
d->chunks[3] = ChunkInit(1 << 3, d->memory.m3);
d->chunks[4] = ChunkInit(1 << 4, d->memory.m4);
d->chunks[5] = ChunkInit(1 << 5, d->memory.m5);
}
static int GetQuantError(unsigned char r, unsigned char g, unsigned char b) {
struct TtyRgb q = tty2rgb(rgb2tty(r, g, b));
return ((r - q.r) + (g - q.g) + (b - q.b)) / 3;
}
static int SerpentineDitherSq2(long yw, long xw, unsigned char rgb[3][yw][xw],
long y, long x, long b, const int ci[b * b][2],
int e) {
long i;
for (i = 0; i < b * b; ++i) {
e = GetQuantError((rgb[0][y + ci[i][0]][x + ci[i][1]] =
CLAMP(rgb[0][y + ci[i][0]][x + ci[i][1]] + e)),
(rgb[1][y + ci[i][0]][x + ci[i][1]] =
CLAMP(rgb[1][y + ci[i][0]][x + ci[i][1]] + e)),
(rgb[2][y + ci[i][0]][x + ci[i][1]] =
CLAMP(rgb[2][y + ci[i][0]][x + ci[i][1]] + e))) *
15 / 16;
}
return e;
}
static void SerpentineDither(long yw, long xw, unsigned char rgb[3][yw][xw],
long yn, long xn, long y, long x, long r,
const struct Dither *d) {
void *c;
long b, e, i, j, n, m;
e = 0;
b = d->chunks[r].b;
c = d->chunks[r].c;
n = (yn - y) / b;
m = (xn - x) / b;
for (i = 0; i < n; ++i) {
for (j = 0; j < m; ++j) {
e = SerpentineDitherSq2(yw, xw, rgb, y + i * b, x + j * b, b, c, 0);
}
}
if (r) {
SerpentineDither(yw, xw, rgb, yn, xn, y + 0 * 0, x + m * b, r - 1, d);
SerpentineDither(yw, xw, rgb, yn, xn, y + n * b, x + 0 * 0, r - 1, d);
SerpentineDither(yw, xw, rgb, yn, xn, y + n * b, x + m * b, r - 1, d);
}
}
/**
* Makes color banding go away a little in low color modes.
*/
void dither(long yw, long xw, unsigned char rgb[3][yw][xw], long yn, long xn) {
static bool once;
if (!once) {
DitherInit(&kDither);
once = true;
}
SerpentineDither(yw, xw, rgb, yn, xn, 0, 0, ARRAYLEN(kDither.chunks) - 1,
&kDither);
}