125 lines
4.9 KiB
C
125 lines
4.9 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 │
|
||
|
│ │
|
||
|
│ This program is free software; you can redistribute it and/or modify │
|
||
|
│ it under the terms of the GNU General Public License as published by │
|
||
|
│ the Free Software Foundation; version 2 of the License. │
|
||
|
│ │
|
||
|
│ This program is distributed in the hope that it will be useful, but │
|
||
|
│ WITHOUT ANY WARRANTY; without even the implied warranty of │
|
||
|
│ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU │
|
||
|
│ General Public License for more details. │
|
||
|
│ │
|
||
|
│ You should have received a copy of the GNU General Public License │
|
||
|
│ along with this program; if not, write to the Free Software │
|
||
|
│ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA │
|
||
|
│ 02110-1301 USA │
|
||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||
|
#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);
|
||
|
}
|