2023-06-15 13:32:20 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <math.h>
|
2023-06-16 06:11:49 +00:00
|
|
|
#include <stddef.h>
|
2023-06-15 13:32:20 +00:00
|
|
|
#include <assert.h>
|
|
|
|
|
2023-06-16 15:23:07 +00:00
|
|
|
#include <raylib.h>
|
|
|
|
#include <rlgl.h>
|
|
|
|
#include <external/glad.h>
|
2023-06-15 13:32:20 +00:00
|
|
|
#define LIGHTMAPPER_IMPLEMENTATION
|
|
|
|
#define LM_DEBUG_INTERPOLATION
|
|
|
|
#include "lightmapper.h"
|
|
|
|
|
|
|
|
#ifndef M_PI // even with _USE_MATH_DEFINES not always available
|
|
|
|
#define M_PI 3.14159265358979323846
|
|
|
|
#endif
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
float p[3];
|
|
|
|
float t[2];
|
|
|
|
} vertex_t;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
2023-06-16 15:23:07 +00:00
|
|
|
//GLuint program;
|
|
|
|
Shader raylib_shader;
|
2023-06-15 13:32:20 +00:00
|
|
|
GLint u_lightmap;
|
2023-06-16 15:23:07 +00:00
|
|
|
GLint u_mvp;
|
|
|
|
Texture raylib_texture;
|
2023-06-15 13:32:20 +00:00
|
|
|
GLuint lightmap;
|
|
|
|
int w, h;
|
|
|
|
Model raylib_model;
|
2023-06-16 15:23:07 +00:00
|
|
|
Camera camera;
|
2023-06-15 13:32:20 +00:00
|
|
|
} scene_t;
|
|
|
|
|
|
|
|
static int initScene(scene_t *scene);
|
|
|
|
static void drawScene(scene_t *scene, float *view, float *projection);
|
|
|
|
static void destroyScene(scene_t *scene);
|
|
|
|
|
|
|
|
static int bake(scene_t *scene)
|
|
|
|
{
|
|
|
|
lm_context *ctx = lmCreate(
|
|
|
|
64, // hemisphere resolution (power of two, max=512)
|
|
|
|
0.001f, 100.0f, // zNear, zFar of hemisphere cameras
|
|
|
|
1.0f, 1.0f, 1.0f, // background color (white for ambient occlusion)
|
|
|
|
2, 0.01f, // lightmap interpolation threshold (small differences are interpolated rather than sampled)
|
|
|
|
// check debug_interpolation.tga for an overview of sampled (red) vs interpolated (green) pixels.
|
|
|
|
0.0f); // modifier for camera-to-surface distance for hemisphere rendering.
|
|
|
|
// tweak this to trade-off between interpolated normals quality and other artifacts (see declaration).
|
|
|
|
|
|
|
|
if (!ctx)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Error: Could not initialize lightmapper.\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int w = scene->w, h = scene->h;
|
|
|
|
float *data = calloc(w * h * 4, sizeof(float));
|
|
|
|
lmSetTargetLightmap(ctx, data, w, h, 4);
|
|
|
|
|
2023-06-16 06:11:49 +00:00
|
|
|
Mesh m = scene->raylib_model.meshes[0];
|
|
|
|
|
2023-06-15 13:32:20 +00:00
|
|
|
lmSetGeometry(ctx, NULL, // no transformation in this example
|
2023-06-16 06:11:49 +00:00
|
|
|
LM_FLOAT, (unsigned char*)m.vertices, 0,
|
2023-06-16 07:08:18 +00:00
|
|
|
LM_FLOAT , (unsigned char*)m.normals, 0,
|
2023-06-16 06:11:49 +00:00
|
|
|
LM_FLOAT, (unsigned char*)m.texcoords, 0,
|
|
|
|
m.vertexCount, LM_NONE, 0);
|
2023-06-15 13:32:20 +00:00
|
|
|
|
|
|
|
int vp[4];
|
|
|
|
float view[16], projection[16];
|
|
|
|
double lastUpdateTime = 0.0;
|
2023-06-16 15:23:07 +00:00
|
|
|
rlEnableDepthTest();
|
|
|
|
rlDisableColorBlend();
|
2023-06-15 13:32:20 +00:00
|
|
|
while (lmBegin(ctx, vp, view, projection))
|
|
|
|
{
|
|
|
|
// render to lightmapper framebuffer
|
2023-06-16 06:11:49 +00:00
|
|
|
rlViewport(vp[0], vp[1], vp[2], vp[3]);
|
2023-06-16 15:23:07 +00:00
|
|
|
|
2023-06-15 13:32:20 +00:00
|
|
|
drawScene(scene, view, projection);
|
2023-06-15 21:47:49 +00:00
|
|
|
|
2023-06-15 13:32:20 +00:00
|
|
|
// display progress every second (printf is expensive)
|
2023-06-16 06:11:49 +00:00
|
|
|
double time = GetTime();
|
2023-06-15 13:32:20 +00:00
|
|
|
if (time - lastUpdateTime > 1.0)
|
|
|
|
{
|
|
|
|
lastUpdateTime = time;
|
|
|
|
printf("\r%6.2f%%", lmProgress(ctx) * 100.0f);
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
lmEnd(ctx);
|
|
|
|
}
|
2023-06-16 15:23:07 +00:00
|
|
|
rlDisableDepthTest();
|
|
|
|
rlEnableColorBlend();
|
|
|
|
|
2023-06-16 06:11:49 +00:00
|
|
|
//printf("\rFinished baking %d triangles.\n", scene->indexCount / 3);
|
2023-06-15 13:32:20 +00:00
|
|
|
|
|
|
|
lmDestroy(ctx);
|
|
|
|
|
|
|
|
// postprocess texture
|
|
|
|
float *temp = calloc(w * h * 4, sizeof(float));
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
{
|
|
|
|
lmImageDilate(data, temp, w, h, 4);
|
|
|
|
lmImageDilate(temp, data, w, h, 4);
|
|
|
|
}
|
|
|
|
lmImageSmooth(data, temp, w, h, 4);
|
|
|
|
lmImageDilate(temp, data, w, h, 4);
|
|
|
|
lmImagePower(data, w, h, 4, 1.0f / 2.2f, 0x7); // gamma correct color channels
|
|
|
|
free(temp);
|
|
|
|
|
2023-06-16 06:11:49 +00:00
|
|
|
unsigned char *tempub = (unsigned char*)calloc(w * h * 4, sizeof(unsigned char));
|
|
|
|
lmImageFtoUB(data, tempub, w, h, 4, 1.0f);
|
|
|
|
Image im;
|
|
|
|
im.data = tempub;
|
|
|
|
im.format = PIXELFORMAT_UNCOMPRESSED_R8G8B8A8;
|
|
|
|
im.height = h;
|
|
|
|
im.width = w;
|
|
|
|
ExportImage(im,"result.png");
|
|
|
|
|
|
|
|
// upload result
|
|
|
|
rlUnloadTexture(scene->lightmap);
|
|
|
|
scene->lightmap = rlLoadTexture(tempub, w, h, RL_PIXELFORMAT_UNCOMPRESSED_R8G8B8A8, 1);
|
2023-06-16 15:23:07 +00:00
|
|
|
Texture texture;
|
|
|
|
texture.id = scene->lightmap;
|
|
|
|
texture.width = w;
|
|
|
|
texture.height = h;
|
|
|
|
texture.mipmaps = 1;
|
|
|
|
texture.format = PIXELFORMAT_UNCOMPRESSED_R8G8B8A8;
|
|
|
|
scene->raylib_texture = texture;
|
2023-06-16 06:11:49 +00:00
|
|
|
free(tempub);
|
2023-06-16 15:23:07 +00:00
|
|
|
|
|
|
|
scene->raylib_model.materials[0].maps[MATERIAL_MAP_DIFFUSE].texture = texture;
|
2023-06-16 06:11:49 +00:00
|
|
|
|
2023-06-16 07:08:18 +00:00
|
|
|
// // save result to a file
|
|
|
|
// if (lmImageSaveTGAf("result.tga", data, w, h, 4, 1.0f))
|
|
|
|
// printf("Saved result.tga\n");
|
|
|
|
// free(data);
|
2023-06-15 13:32:20 +00:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2023-06-15 21:47:49 +00:00
|
|
|
static int first = 1;
|
|
|
|
|
|
|
|
static void mainLoop(scene_t *scene)
|
2023-06-15 13:32:20 +00:00
|
|
|
{
|
2023-06-15 21:47:49 +00:00
|
|
|
if (first){
|
2023-06-15 13:32:20 +00:00
|
|
|
bake(scene);
|
2023-06-15 21:47:49 +00:00
|
|
|
first = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int w = GetScreenWidth() * GetWindowScaleDPI().x;
|
|
|
|
int h = GetScreenHeight() * GetWindowScaleDPI().y;
|
2023-06-15 13:32:20 +00:00
|
|
|
|
2023-06-16 15:23:07 +00:00
|
|
|
rlViewport(0, 0, w, h);
|
2023-06-15 13:32:20 +00:00
|
|
|
|
|
|
|
// camera for glfw window
|
|
|
|
float view[16], projection[16];
|
2023-06-16 15:23:07 +00:00
|
|
|
//fpsCameraViewMatrix(view);
|
|
|
|
//perspectiveMatrix(projection, 45.0f, (float)w / (float)h, 0.01f, 100.0f);
|
2023-06-15 13:32:20 +00:00
|
|
|
|
2023-06-15 21:47:49 +00:00
|
|
|
BeginDrawing();
|
2023-06-16 15:23:07 +00:00
|
|
|
BeginMode3D(scene->camera);
|
|
|
|
ClearBackground(BLUE);
|
|
|
|
rlEnableDepthTest();
|
|
|
|
//drawScene(scene, view, projection);
|
|
|
|
DrawModel(scene->raylib_model, (Vector3) {0.0f,0.0f,0.0f}, 1, WHITE);
|
|
|
|
rlDisableDepthTest();
|
|
|
|
EndMode3D();
|
2023-06-15 21:47:49 +00:00
|
|
|
EndDrawing();
|
2023-06-15 13:32:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char* argv[])
|
|
|
|
{
|
2023-06-15 21:47:49 +00:00
|
|
|
SetConfigFlags(FLAG_MSAA_4X_HINT | FLAG_WINDOW_HIGHDPI | FLAG_VSYNC_HINT);
|
2023-06-15 13:32:20 +00:00
|
|
|
|
2023-06-15 21:47:49 +00:00
|
|
|
InitWindow(1024,768,"Test");
|
2023-06-15 13:32:20 +00:00
|
|
|
|
|
|
|
scene_t scene = {0};
|
|
|
|
if (!initScene(&scene))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Could not initialize scene.\n");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("Ambient Occlusion Baking Example.\n");
|
|
|
|
printf("Use your mouse and the W, A, S, D, E, Q keys to navigate.\n");
|
|
|
|
printf("Press SPACE to start baking one light bounce!\n");
|
|
|
|
printf("This will take a few seconds and bake a lightmap illuminated by:\n");
|
|
|
|
printf("1. The mesh itself (initially black)\n");
|
|
|
|
printf("2. A white sky (1.0f, 1.0f, 1.0f)\n");
|
|
|
|
|
2023-06-15 21:47:49 +00:00
|
|
|
while (!WindowShouldClose())
|
2023-06-15 13:32:20 +00:00
|
|
|
{
|
2023-06-15 21:47:49 +00:00
|
|
|
mainLoop(&scene);
|
2023-06-15 13:32:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
destroyScene(&scene);
|
2023-06-15 21:47:49 +00:00
|
|
|
CloseWindow();
|
2023-06-15 13:32:20 +00:00
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int initScene(scene_t *scene)
|
|
|
|
{
|
|
|
|
// load mesh
|
2023-06-16 06:11:49 +00:00
|
|
|
scene->raylib_model = LoadModel("thirdparty/lightmapper/example/gazebo.obj");
|
|
|
|
Mesh m = scene->raylib_model.meshes[0];
|
|
|
|
if(m.normals != NULL) puts("Has normals");
|
|
|
|
if(m.texcoords != NULL) puts("Has texcoords");
|
|
|
|
if(m.texcoords2 != NULL) puts("Has texcoords2");
|
2023-06-16 15:23:07 +00:00
|
|
|
if(m.indices != NULL) puts("Has indices");
|
|
|
|
|
|
|
|
scene->w = 512;
|
|
|
|
scene->h = 512;
|
|
|
|
scene->lightmap = LoadTextureFromImage(GenImageColor(1,1,BLACK)).id;
|
2023-06-15 13:32:20 +00:00
|
|
|
|
|
|
|
// load shader
|
|
|
|
const char *vp =
|
|
|
|
"#version 150 core\n"
|
2023-06-16 15:23:07 +00:00
|
|
|
"in vec3 vertexPosition;\n"
|
|
|
|
"in vec2 vertexTexCoord;\n"
|
|
|
|
"uniform mat4 mvp;\n"
|
2023-06-15 13:32:20 +00:00
|
|
|
"out vec2 v_texcoord;\n"
|
|
|
|
|
|
|
|
"void main()\n"
|
|
|
|
"{\n"
|
2023-06-16 15:23:07 +00:00
|
|
|
"gl_Position = mvp * vec4(vertexPosition, 1.0);\n"
|
|
|
|
"v_texcoord = vertexTexCoord;\n"
|
2023-06-15 13:32:20 +00:00
|
|
|
"}\n";
|
|
|
|
|
|
|
|
const char *fp =
|
|
|
|
"#version 150 core\n"
|
|
|
|
"in vec2 v_texcoord;\n"
|
2023-06-16 15:23:07 +00:00
|
|
|
"uniform sampler2D texture0;\n"
|
2023-06-15 13:32:20 +00:00
|
|
|
"out vec4 o_color;\n"
|
|
|
|
|
|
|
|
"void main()\n"
|
|
|
|
"{\n"
|
2023-06-16 15:23:07 +00:00
|
|
|
"o_color = vec4(texture(texture0, v_texcoord).rgb, gl_FrontFacing ? 1.0 : 0.0);\n"
|
2023-06-15 13:32:20 +00:00
|
|
|
"}\n";
|
|
|
|
|
2023-06-16 15:23:07 +00:00
|
|
|
//scene->program = rlLoadShaderCode(vp, fp);
|
|
|
|
scene->raylib_shader = LoadShaderFromMemory(vp, fp);
|
2023-06-15 13:32:20 +00:00
|
|
|
|
2023-06-16 15:23:07 +00:00
|
|
|
scene->u_lightmap = rlGetLocationUniform(scene->raylib_shader.id, "texture0");
|
|
|
|
scene->u_mvp = rlGetLocationUniform(scene->raylib_shader.id, "mvp");
|
|
|
|
|
|
|
|
//scene->raylib_model.materials[0].shader = scene->raylib_shader;
|
|
|
|
|
|
|
|
Camera camera = { 0 };
|
|
|
|
camera.position = (Vector3){ 1.0f, 0.5f, 1.0f }; // Camera position
|
|
|
|
camera.target = (Vector3){ 0.0f, 0.0f, 0.0f }; // Camera looking at point
|
|
|
|
camera.up = (Vector3){ 0.0f, 1.0f, 0.0f }; // Camera up vector (rotation towards target)
|
|
|
|
camera.fovy = 45.0f; // Camera field-of-view Y
|
|
|
|
camera.projection = CAMERA_PERSPECTIVE; // Camera mode type
|
|
|
|
|
|
|
|
scene->camera = camera;
|
2023-06-15 13:32:20 +00:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2023-06-16 15:23:07 +00:00
|
|
|
static void multiplyMatrices(float *out, float *a, float *b)
|
|
|
|
{
|
|
|
|
for (int y = 0; y < 4; y++)
|
|
|
|
for (int x = 0; x < 4; x++)
|
|
|
|
out[y * 4 + x] = a[x] * b[y * 4] + a[4 + x] * b[y * 4 + 1] + a[8 + x] * b[y * 4 + 2] + a[12 + x] * b[y * 4 + 3];
|
|
|
|
}
|
|
|
|
|
2023-06-15 13:32:20 +00:00
|
|
|
static void drawScene(scene_t *scene, float *view, float *projection)
|
|
|
|
{
|
2023-06-16 06:11:49 +00:00
|
|
|
Mesh m = scene->raylib_model.meshes[0];
|
2023-06-15 13:32:20 +00:00
|
|
|
|
2023-06-16 15:23:07 +00:00
|
|
|
rlEnableShader(scene->raylib_shader.id);
|
|
|
|
float mvp[16];
|
|
|
|
multiplyMatrices(mvp, projection, view);
|
|
|
|
glUniformMatrix4fv(scene->u_mvp, 1, GL_FALSE, mvp);
|
2023-06-15 13:32:20 +00:00
|
|
|
|
|
|
|
glBindTexture(GL_TEXTURE_2D, scene->lightmap);
|
|
|
|
|
2023-06-16 06:11:49 +00:00
|
|
|
//glBindVertexArray(scene->vao);
|
|
|
|
//glDrawElements(GL_TRIANGLES, scene->indexCount, GL_UNSIGNED_SHORT, 0);
|
|
|
|
glBindVertexArray(m.vaoId);
|
|
|
|
glDrawArrays(GL_TRIANGLES, 0, m.vertexCount);
|
2023-06-15 13:32:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void destroyScene(scene_t *scene)
|
|
|
|
{
|
2023-06-16 15:23:07 +00:00
|
|
|
UnloadModel(scene->raylib_model);
|
|
|
|
UnloadTexture(scene->raylib_texture);
|
|
|
|
UnloadShader(scene->raylib_shader);
|
|
|
|
}
|