213 lines
7.6 KiB
C
213 lines
7.6 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 "libc/assert.h"
|
||
|
#include "libc/calls/calls.h"
|
||
|
#include "libc/calls/struct/stat.h"
|
||
|
#include "libc/limits.h"
|
||
|
#include "libc/log/check.h"
|
||
|
#include "libc/macros.h"
|
||
|
#include "libc/math.h"
|
||
|
#include "libc/mem/mem.h"
|
||
|
#include "libc/stdio/stdio.h"
|
||
|
#include "libc/str/str.h"
|
||
|
#include "libc/sysv/consts/madv.h"
|
||
|
#include "libc/sysv/consts/map.h"
|
||
|
#include "libc/sysv/consts/o.h"
|
||
|
#include "libc/sysv/consts/prot.h"
|
||
|
#include "libc/x/x.h"
|
||
|
#include "third_party/stb/stb_image.h"
|
||
|
|
||
|
#define C13(A, B) (((A) + 3 * (B)) / 4)
|
||
|
#define S3T(A, B, C) MAX(0, ((24 * (B)) - (4 * ((A) + (C)))) / 16)
|
||
|
#define LERP(A, B, P) ((A) * (1 - (P)) + (B) * (P))
|
||
|
|
||
|
float ByteToFloat(int b) {
|
||
|
return 1 / 255.f * b;
|
||
|
}
|
||
|
|
||
|
int FloatToByte(float f) {
|
||
|
return MAX(0, MIN(255, roundf(255 * f)));
|
||
|
}
|
||
|
|
||
|
float ChessBoard(unsigned y, unsigned x, float a, float b) {
|
||
|
return !((y ^ x) & (1u << 2)) ? a : b;
|
||
|
}
|
||
|
|
||
|
float AlphaBackground(unsigned y, unsigned x) {
|
||
|
return ChessBoard(y, x, 1.f, .7f);
|
||
|
}
|
||
|
|
||
|
float OutOfBoundsBackground(unsigned y, unsigned x) {
|
||
|
return ChessBoard(y, x, .01f, .02f);
|
||
|
}
|
||
|
|
||
|
float Opacify(size_t yn, size_t xn, const float P[yn][xn],
|
||
|
const float A[yn][xn], long y, long x) {
|
||
|
if ((0 <= y && y < yn) && (0 <= x && x < xn)) {
|
||
|
return LERP(AlphaBackground(y, x), P[y][x], A[y][x]);
|
||
|
} else {
|
||
|
return OutOfBoundsBackground(y, x);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void PrintImage(size_t yn, size_t xn, float R[yn][xn], float G[yn][xn],
|
||
|
float B[yn][xn], float A[yn][xn]) {
|
||
|
unsigned y, x;
|
||
|
for (y = 0; y < yn; y += 2) {
|
||
|
if (y) printf("\e[0m\n");
|
||
|
for (x = 0; x < xn; ++x) {
|
||
|
printf("\e[48;2;%d;%d;%d;38;2;%d;%d;%dm▄",
|
||
|
FloatToByte(Opacify(yn, xn, R, A, y + 0, x)),
|
||
|
FloatToByte(Opacify(yn, xn, G, A, y + 0, x)),
|
||
|
FloatToByte(Opacify(yn, xn, B, A, y + 0, x)),
|
||
|
FloatToByte(Opacify(yn, xn, R, A, y + 1, x)),
|
||
|
FloatToByte(Opacify(yn, xn, G, A, y + 1, x)),
|
||
|
FloatToByte(Opacify(yn, xn, B, A, y + 1, x)));
|
||
|
}
|
||
|
}
|
||
|
printf("\e[0m\n");
|
||
|
}
|
||
|
|
||
|
void DeblinterlaceRgba(size_t dyn, size_t dxn, float R[dyn][dxn],
|
||
|
float G[dyn][dxn], float B[dyn][dxn], float A[dyn][dxn],
|
||
|
size_t syn, size_t sxn,
|
||
|
const unsigned char src[syn][sxn][4]) {
|
||
|
unsigned y, x;
|
||
|
for (y = 0; y < syn; ++y) {
|
||
|
for (x = 0; x < sxn; ++x) {
|
||
|
R[y][x] = ByteToFloat(src[y][x][0]);
|
||
|
G[y][x] = ByteToFloat(src[y][x][1]);
|
||
|
B[y][x] = ByteToFloat(src[y][x][2]);
|
||
|
A[y][x] = ByteToFloat(src[y][x][3]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void SharpenX(size_t yw, size_t xw, float dst[yw][xw], const float src[yw][xw],
|
||
|
size_t yn, size_t xn) {
|
||
|
int y, x;
|
||
|
for (y = 0; y < yn; ++y) {
|
||
|
for (x = 0; x < xn; ++x) {
|
||
|
dst[y][x] =
|
||
|
S3T(src[y][MAX(0, x - 1)], src[y][x], src[y][MIN(xn - 1, x + 1)]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void SharpenY(size_t yw, size_t xw, float dst[yw][xw], const float src[yw][xw],
|
||
|
size_t yn, size_t xn) {
|
||
|
int y, x;
|
||
|
for (y = 0; y < yn; ++y) {
|
||
|
for (x = 0; x < xn; ++x) {
|
||
|
dst[y][x] =
|
||
|
S3T(src[MAX(0, y - 1)][x], src[y][x], src[MIN(yn - 1, y + 1)][x]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void UpscaleX(size_t yw, size_t xw, float img[yw][xw], size_t yn, size_t xn) {
|
||
|
unsigned y, x;
|
||
|
for (y = yn; y--;) {
|
||
|
for (x = xn; --x;) {
|
||
|
img[y][x] =
|
||
|
C13(img[y][MIN(xn / 2 - 1, x / 2 - 1 + x % 2 * 2)], img[y][x / 2]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void UpscaleY(size_t yw, size_t xw, float img[yw][xw], size_t yn, size_t xn) {
|
||
|
unsigned y, x;
|
||
|
for (y = yn; --y;) {
|
||
|
for (x = xn; x--;) {
|
||
|
img[y][x] =
|
||
|
C13(img[MIN(yn / 2 - 1, y / 2 - 1 + y % 2 * 2)][x], img[y / 2][x]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Upscale(size_t yw, size_t xw, float img[yw][xw], float tmp[yw][xw],
|
||
|
size_t yn, size_t xn) {
|
||
|
UpscaleY(yw, xw, img, yn, xn / 2);
|
||
|
SharpenY(yw, xw, tmp, img, yn, xn / 2);
|
||
|
UpscaleX(yw, xw, tmp, yn, xn);
|
||
|
SharpenX(yw, xw, img, tmp, yn, xn);
|
||
|
}
|
||
|
|
||
|
void ProcessImageDouble(size_t yn, size_t xn, unsigned char img[yn][xn][4]) {
|
||
|
void *t = xmemalign(32, sizeof(float) * yn * 2 * xn * 2);
|
||
|
void *R = xmemalign(32, sizeof(float) * yn * 2 * xn * 2);
|
||
|
void *G = xmemalign(32, sizeof(float) * yn * 2 * xn * 2);
|
||
|
void *B = xmemalign(32, sizeof(float) * yn * 2 * xn * 2);
|
||
|
void *A = xmemalign(32, sizeof(float) * yn * 2 * xn * 2);
|
||
|
DeblinterlaceRgba(yn * 2, xn * 2, R, G, B, A, yn, xn, img);
|
||
|
Upscale(yn * 2, xn * 2, R, t, yn * 2, xn * 2);
|
||
|
Upscale(yn * 2, xn * 2, G, t, yn * 2, xn * 2);
|
||
|
Upscale(yn * 2, xn * 2, B, t, yn * 2, xn * 2);
|
||
|
Upscale(yn * 2, xn * 2, A, t, yn * 2, xn * 2);
|
||
|
free(t);
|
||
|
PrintImage(yn * 2, xn * 2, R, G, B, A);
|
||
|
free(R);
|
||
|
free(G);
|
||
|
free(B);
|
||
|
free(A);
|
||
|
}
|
||
|
|
||
|
void ProcessImage(size_t yn, size_t xn, unsigned char img[yn][xn][4]) {
|
||
|
void *R = xmemalign(32, sizeof(float) * yn * xn);
|
||
|
void *G = xmemalign(32, sizeof(float) * yn * xn);
|
||
|
void *B = xmemalign(32, sizeof(float) * yn * xn);
|
||
|
void *A = xmemalign(32, sizeof(float) * yn * xn);
|
||
|
DeblinterlaceRgba(yn, xn, R, G, B, A, yn, xn, img);
|
||
|
PrintImage(yn, xn, R, G, B, A);
|
||
|
free(R);
|
||
|
free(G);
|
||
|
free(B);
|
||
|
free(A);
|
||
|
}
|
||
|
|
||
|
void WithImageFile(const char *path, void fn(size_t yn, size_t xn,
|
||
|
unsigned char img[yn][xn][4])) {
|
||
|
struct stat st;
|
||
|
int fd, yn, xn;
|
||
|
void *map, *data;
|
||
|
CHECK_NE(-1, (fd = open(path, O_RDONLY)), "%s", path);
|
||
|
CHECK_NE(-1, fstat(fd, &st));
|
||
|
CHECK_GT(st.st_size, 0);
|
||
|
CHECK_LE(st.st_size, INT_MAX);
|
||
|
fadvise(fd, 0, 0, MADV_WILLNEED | MADV_SEQUENTIAL);
|
||
|
CHECK_NE(MAP_FAILED,
|
||
|
(map = mmap(NULL, st.st_size, PROT_READ, MAP_SHARED, fd, 0)));
|
||
|
CHECK_NOTNULL(
|
||
|
(data = stbi_load_from_memory(map, st.st_size, &xn, &yn, NULL, 4)), "%s",
|
||
|
path);
|
||
|
CHECK_NE(-1, munmap(map, st.st_size));
|
||
|
CHECK_NE(-1, close(fd));
|
||
|
fn(yn, xn, data);
|
||
|
free(data);
|
||
|
}
|
||
|
|
||
|
int main(int argc, char *argv[]) {
|
||
|
int i;
|
||
|
for (i = 1; i < argc; ++i) {
|
||
|
WithImageFile(argv[i], ProcessImageDouble);
|
||
|
}
|
||
|
return 0;
|
||
|
}
|