tinyrenderer/src/img/img.c
Abdelrahman 50b8c6dd0a Implement shaders (#1)
Reviewed-on: #1

	* Start implementing shaders and move vector code to dedicated files
	* Extract shader into its own header
	* Ensure pixel coordinates are within bounds
	* Refactor shader code and implement fragment shaders
	* Create shaders
	* Reorganise code
Co-authored-by: Abdelrahman <said.abdelrahman89@gmail.com>
Co-committed-by: Abdelrahman <said.abdelrahman89@gmail.com>
2024-08-18 14:28:09 +00:00

94 lines
2.3 KiB
C

#include "img.h"
#include "aliases.h"
#include "mem_arena.h"
#include "pam.h"
#include "utils.h"
#include <stdbool.h>
#include <stddef.h>
#include <string.h>
internal i64 calculate_pixel_index(Buffer *buffer, u64 x, u64 y, u64 base_size);
bool _init_buffer(Arena *arena, Buffer *buffer, u64 base_size) {
if (!arena || !buffer || buffer->width == 0 || buffer->height == 0) {
return false;
}
u64 size = buffer->width * buffer->height * base_size;
buffer->buf = wapp_mem_arena_alloc(arena, size);
return buffer->buf != NULL;
}
u8 *_get_pixel(Buffer *buffer, u64 x, u64 y, u64 base_size) {
i64 idx = calculate_pixel_index(buffer, x, y, base_size);
if (idx == -1) {
idx = 0;
}
return ((u8 *)(buffer->buf)) + idx;
}
void _set_pixel(Buffer *buffer, u64 x, u64 y, void *value, u64 base_size) {
i64 idx = calculate_pixel_index(buffer, x, y, base_size);
if (idx == -1) {
return;
}
memcpy(((u8 *)(buffer->buf)) + idx, value, base_size);
return;
}
void _clear_buffer(Buffer *buffer, void *value, u64 base_size) {
for (u64 y = 0; y < buffer->height; ++y) {
for (u64 x = 0; x < buffer->width; ++x) {
_set_pixel(buffer, x, y, value, base_size);
}
}
}
void draw_line(Image *img, u64 x0, u64 y0, u64 x1, u64 y1, Colour colour) {
u64 dx = absolute((i64)x1 - (i64)x0);
u64 dy = absolute((i64)y1 - (i64)y0);
f32 t = 0.0f;
i64 x = 0;
i64 y = 0;
if (dx > dy) {
if (x0 > x1) {
swap(u64, x0, x1);
swap(u64, y0, y1);
}
for (u64 x = x0; x < x1; ++x) {
t = (f32)(x - x0) / (f32)(x1 - x0);
y = (i64)y0 + ((i64)y1 - (i64)y0) * t;
set_pixel(img, x, y, &colour);
}
} else {
if (y0 > y1) {
swap(u64, x0, x1);
swap(u64, y0, y1);
}
for (u64 y = y0; y < y1; ++y) {
t = (f32)(y - y0) / (f32)(y1 - y0);
x = (i64)x0 + ((i64)x1 - (i64)x0) * t;
set_pixel(img, x, y, &colour);
}
}
}
void save_image(const Image *img, const char *filename) {
write_p7_image(img->width, img->height, (u8 *)(img->buf), filename);
}
internal i64 calculate_pixel_index(Buffer *buffer, u64 x, u64 y,
u64 base_size) {
if (x < 0 || y < 0 || x >= buffer->width || y >= buffer->height) {
return -1;
}
return (y * buffer->width + x) * base_size;
}