1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
|
#include "skybox.h"
#include <assert.h>
#include "file.h"
#include "glad/glad.h"
#include "log.h"
#include "primitives.h"
#include "ral_common.h"
#include "ral_impl.h"
#include "ral_types.h"
#include "render.h"
#include "render_types.h"
Skybox Skybox_Create(const char** face_paths, int n) {
INFO("Creating a skybox");
assert(n == 6); // ! we're only supporting a full cubemap for now
// -- cube verts
Geometry geom = Geo_CreateCuboid(f32x3(1.0, 1.0, 1.0));
Mesh cube = Mesh_Create(&geom, true);
// -- cubemap texture
TextureHandle handle;
GPU_Texture* tex = GPU_TextureAlloc(&handle);
glBindTexture(GL_TEXTURE_CUBE_MAP, tex->id);
int width, height, nrChannels;
// unsigned char *data;
for (unsigned int i = 0; i < n; i++) {
TextureData data = TextureDataLoad(
face_paths[i],
false); // stbi_load(textures_faces[i].c_str(), &width, &height, &nrChannels, 0);
assert(data.description.format == TEXTURE_FORMAT_8_8_8_RGB_UNORM);
glTexImage2D(GL_TEXTURE_CUBE_MAP_POSITIVE_X + i, 0, GL_RGB, data.description.extents.x,
data.description.extents.y, 0, GL_RGB, GL_UNSIGNED_BYTE, data.image_data);
}
glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_R, GL_CLAMP_TO_EDGE);
// shader pipeline
ShaderData shader_data = { .data = NULL, .get_layout = &Skybox_GetLayout };
GPU_RenderpassDesc rpass_desc = {
.default_framebuffer = true,
};
GPU_Renderpass* pass = GPU_Renderpass_Create(rpass_desc);
arena scratch = arena_create(malloc(1024 * 1024), 1024 * 1024);
Str8 vert_path = str8("assets/shaders/skybox.vert");
Str8 frag_path = str8("assets/shaders/pbr_textured.frag");
str8_opt vertex_shader = str8_from_file(&scratch, vert_path);
str8_opt fragment_shader = str8_from_file(&scratch, frag_path);
if (!vertex_shader.has_value || !fragment_shader.has_value) {
ERROR_EXIT("Failed to load shaders from disk")
}
VertexDescription pos_only = { .debug_label = "Position only verts" };
VertexDesc_AddAttr(&pos_only, "inPos", ATTR_F32x3);
pos_only.use_full_vertex_size = true;
GraphicsPipelineDesc pipeline_desc = {
.debug_name = "Skybox pipeline",
.vertex_desc = pos_only,
.data_layouts = { shader_data },
.data_layouts_count = 1,
.vs = {
.debug_name = "Skybox Vertex Shader",
.filepath = vert_path,
.code = vertex_shader.contents
},
.fs = {
.debug_name = "Skybox Fragment Shader",
.filepath = frag_path,
.code = fragment_shader.contents
},
.wireframe = false,
.depth_test = true,
};
GPU_Pipeline* pipeline = GPU_GraphicsPipeline_Create(pipeline_desc, pass);
return (Skybox){ .cube = cube, .texture = handle, .pipeline = pipeline };
}
void Skybox_Draw(Skybox* skybox, Camera camera) {
GPU_CmdEncoder* enc = GPU_GetDefaultEncoder();
GPU_CmdEncoder_BeginRender(enc, skybox->pipeline->renderpass);
GPU_EncodeBindPipeline(enc, skybox->pipeline);
GPU_EncodeSetDefaults(enc);
// Shader data
SkyboxUniforms uniforms = { .in_position = camera.position, .cubemap = skybox->texture };
ShaderData skybox_data = { .data = &uniforms, .get_layout = Skybox_GetLayout };
GPU_EncodeSetVertexBuffer(enc, skybox->cube.vertex_buffer);
GPU_EncodeSetVertexBuffer(enc, skybox->cube.index_buffer);
GPU_EncodeDrawIndexed(enc, skybox->cube.geometry->indices->len);
GPU_CmdEncoder_EndRender(enc);
}
|