249 lines
7.8 KiB
C
249 lines
7.8 KiB
C
#include <3ds.h>
|
|
#include <citro3d.h>
|
|
#include <string.h>
|
|
#include "vshader_shbin.h"
|
|
|
|
#define CLEAR_COLOR 0x68B0D8FF
|
|
|
|
#define DISPLAY_TRANSFER_FLAGS \
|
|
(GX_TRANSFER_FLIP_VERT(0) | GX_TRANSFER_OUT_TILED(0) | GX_TRANSFER_RAW_COPY(0) | \
|
|
GX_TRANSFER_IN_FORMAT(GX_TRANSFER_FMT_RGBA8) | GX_TRANSFER_OUT_FORMAT(GX_TRANSFER_FMT_RGB8) | \
|
|
GX_TRANSFER_SCALING(GX_TRANSFER_SCALE_NO))
|
|
|
|
typedef struct { float position[3]; float texcoord[2]; float normal[3]; } vertex;
|
|
|
|
static const vertex vertex_list[] =
|
|
{
|
|
// First face (PZ)
|
|
// First triangle
|
|
{ {-0.5f, -0.5f, +0.5f}, {0.0f, 0.0f}, {0.0f, 0.0f, +1.0f} },
|
|
{ {+0.5f, -0.5f, +0.5f}, {1.0f, 0.0f}, {0.0f, 0.0f, +1.0f} },
|
|
{ {+0.5f, +0.5f, +0.5f}, {1.0f, 1.0f}, {0.0f, 0.0f, +1.0f} },
|
|
// Second triangle
|
|
{ {+0.5f, +0.5f, +0.5f}, {1.0f, 1.0f}, {0.0f, 0.0f, +1.0f} },
|
|
{ {-0.5f, +0.5f, +0.5f}, {0.0f, 1.0f}, {0.0f, 0.0f, +1.0f} },
|
|
{ {-0.5f, -0.5f, +0.5f}, {0.0f, 0.0f}, {0.0f, 0.0f, +1.0f} },
|
|
|
|
// Second face (MZ)
|
|
// First triangle
|
|
{ {-0.5f, -0.5f, -0.5f}, {0.0f, 0.0f}, {0.0f, 0.0f, -1.0f} },
|
|
{ {-0.5f, +0.5f, -0.5f}, {1.0f, 0.0f}, {0.0f, 0.0f, -1.0f} },
|
|
{ {+0.5f, +0.5f, -0.5f}, {1.0f, 1.0f}, {0.0f, 0.0f, -1.0f} },
|
|
// Second triangle
|
|
{ {+0.5f, +0.5f, -0.5f}, {1.0f, 1.0f}, {0.0f, 0.0f, -1.0f} },
|
|
{ {+0.5f, -0.5f, -0.5f}, {0.0f, 1.0f}, {0.0f, 0.0f, -1.0f} },
|
|
{ {-0.5f, -0.5f, -0.5f}, {0.0f, 0.0f}, {0.0f, 0.0f, -1.0f} },
|
|
|
|
// Third face (PX)
|
|
// First triangle
|
|
{ {+0.5f, -0.5f, -0.5f}, {0.0f, 0.0f}, {+1.0f, 0.0f, 0.0f} },
|
|
{ {+0.5f, +0.5f, -0.5f}, {1.0f, 0.0f}, {+1.0f, 0.0f, 0.0f} },
|
|
{ {+0.5f, +0.5f, +0.5f}, {1.0f, 1.0f}, {+1.0f, 0.0f, 0.0f} },
|
|
// Second triangle
|
|
{ {+0.5f, +0.5f, +0.5f}, {1.0f, 1.0f}, {+1.0f, 0.0f, 0.0f} },
|
|
{ {+0.5f, -0.5f, +0.5f}, {0.0f, 1.0f}, {+1.0f, 0.0f, 0.0f} },
|
|
{ {+0.5f, -0.5f, -0.5f}, {0.0f, 0.0f}, {+1.0f, 0.0f, 0.0f} },
|
|
|
|
// Fourth face (MX)
|
|
// First triangle
|
|
{ {-0.5f, -0.5f, -0.5f}, {0.0f, 0.0f}, {-1.0f, 0.0f, 0.0f} },
|
|
{ {-0.5f, -0.5f, +0.5f}, {1.0f, 0.0f}, {-1.0f, 0.0f, 0.0f} },
|
|
{ {-0.5f, +0.5f, +0.5f}, {1.0f, 1.0f}, {-1.0f, 0.0f, 0.0f} },
|
|
// Second triangle
|
|
{ {-0.5f, +0.5f, +0.5f}, {1.0f, 1.0f}, {-1.0f, 0.0f, 0.0f} },
|
|
{ {-0.5f, +0.5f, -0.5f}, {0.0f, 1.0f}, {-1.0f, 0.0f, 0.0f} },
|
|
{ {-0.5f, -0.5f, -0.5f}, {0.0f, 0.0f}, {-1.0f, 0.0f, 0.0f} },
|
|
|
|
// Fifth face (PY)
|
|
// First triangle
|
|
{ {-0.5f, +0.5f, -0.5f}, {0.0f, 0.0f}, {0.0f, +1.0f, 0.0f} },
|
|
{ {-0.5f, +0.5f, +0.5f}, {1.0f, 0.0f}, {0.0f, +1.0f, 0.0f} },
|
|
{ {+0.5f, +0.5f, +0.5f}, {1.0f, 1.0f}, {0.0f, +1.0f, 0.0f} },
|
|
// Second triangle
|
|
{ {+0.5f, +0.5f, +0.5f}, {1.0f, 1.0f}, {0.0f, +1.0f, 0.0f} },
|
|
{ {+0.5f, +0.5f, -0.5f}, {0.0f, 1.0f}, {0.0f, +1.0f, 0.0f} },
|
|
{ {-0.5f, +0.5f, -0.5f}, {0.0f, 0.0f}, {0.0f, +1.0f, 0.0f} },
|
|
|
|
// Sixth face (MY)
|
|
// First triangle
|
|
{ {-0.5f, -0.5f, -0.5f}, {0.0f, 0.0f}, {0.0f, -1.0f, 0.0f} },
|
|
{ {+0.5f, -0.5f, -0.5f}, {1.0f, 0.0f}, {0.0f, -1.0f, 0.0f} },
|
|
{ {+0.5f, -0.5f, +0.5f}, {1.0f, 1.0f}, {0.0f, -1.0f, 0.0f} },
|
|
// Second triangle
|
|
{ {+0.5f, -0.5f, +0.5f}, {1.0f, 1.0f}, {0.0f, -1.0f, 0.0f} },
|
|
{ {-0.5f, -0.5f, +0.5f}, {0.0f, 1.0f}, {0.0f, -1.0f, 0.0f} },
|
|
{ {-0.5f, -0.5f, -0.5f}, {0.0f, 0.0f}, {0.0f, -1.0f, 0.0f} },
|
|
};
|
|
|
|
#define vertex_list_count (sizeof(vertex_list)/sizeof(vertex_list[0]))
|
|
|
|
static DVLB_s* vshader_dvlb;
|
|
static shaderProgram_s program;
|
|
static int uLoc_projection, uLoc_modelView;
|
|
static C3D_Mtx projection;
|
|
|
|
static C3D_LightEnv lightEnv;
|
|
static C3D_Light light;
|
|
static C3D_LightLut lut_Phong;
|
|
|
|
static void* vbo_data;
|
|
static float angleX = 0.0, angleY = 0.0;
|
|
|
|
static void sceneInit(void)
|
|
{
|
|
// Load the vertex shader, create a shader program and bind it
|
|
vshader_dvlb = DVLB_ParseFile((u32*)vshader_shbin, vshader_shbin_size);
|
|
shaderProgramInit(&program);
|
|
shaderProgramSetVsh(&program, &vshader_dvlb->DVLE[0]);
|
|
C3D_BindProgram(&program);
|
|
|
|
// Get the location of the uniforms
|
|
uLoc_projection = shaderInstanceGetUniformLocation(program.vertexShader, "projection");
|
|
uLoc_modelView = shaderInstanceGetUniformLocation(program.vertexShader, "modelView");
|
|
|
|
// Configure attributes for use with the vertex shader
|
|
C3D_AttrInfo* attrInfo = C3D_GetAttrInfo();
|
|
AttrInfo_Init(attrInfo);
|
|
AttrInfo_AddLoader(attrInfo, 0, GPU_FLOAT, 3); // v0=position
|
|
AttrInfo_AddLoader(attrInfo, 1, GPU_FLOAT, 2); // v1=texcoord
|
|
AttrInfo_AddLoader(attrInfo, 2, GPU_FLOAT, 3); // v2=normal
|
|
|
|
// Create the VBO (vertex buffer object)
|
|
vbo_data = linearAlloc(sizeof(vertex_list));
|
|
memcpy(vbo_data, vertex_list, sizeof(vertex_list));
|
|
|
|
// Configure buffers
|
|
C3D_BufInfo* bufInfo = C3D_GetBufInfo();
|
|
BufInfo_Init(bufInfo);
|
|
BufInfo_Add(bufInfo, vbo_data, sizeof(vertex), 3, 0x210);
|
|
|
|
// Configure the first fragment shading substage to blend the fragment primary color
|
|
// with the fragment secondary color.
|
|
// See https://www.opengl.org/sdk/docs/man2/xhtml/glTexEnv.xml for more insight
|
|
C3D_TexEnv* env = C3D_GetTexEnv(0);
|
|
C3D_TexEnvSrc(env, C3D_Both, GPU_FRAGMENT_PRIMARY_COLOR, GPU_FRAGMENT_SECONDARY_COLOR, 0);
|
|
C3D_TexEnvOp(env, C3D_Both, 0, 0, 0);
|
|
C3D_TexEnvFunc(env, C3D_Both, GPU_ADD);
|
|
|
|
static const C3D_Material material =
|
|
{
|
|
{ 0.2f, 0.2f, 0.2f }, //ambient
|
|
{ 0.4f, 0.4f, 0.4f }, //diffuse
|
|
{ 0.8f, 0.8f, 0.8f }, //specular0
|
|
{ 0.0f, 0.0f, 0.0f }, //specular1
|
|
{ 0.0f, 0.0f, 0.0f }, //emission
|
|
};
|
|
|
|
C3D_LightEnvInit(&lightEnv);
|
|
C3D_LightEnvBind(&lightEnv);
|
|
C3D_LightEnvMaterial(&lightEnv, &material);
|
|
|
|
LightLut_Phong(&lut_Phong, 30);
|
|
C3D_LightEnvLut(&lightEnv, GPU_LUT_D0, GPU_LUTINPUT_LN, false, &lut_Phong);
|
|
|
|
C3D_FVec lightVec = { { 1.0, -0.5, 0.0, 0.0 } };
|
|
|
|
C3D_LightInit(&light, &lightEnv);
|
|
C3D_LightColor(&light, 1.0, 1.0, 1.0);
|
|
C3D_LightPosition(&light, &lightVec);
|
|
}
|
|
|
|
static void sceneRender(float iod)
|
|
{
|
|
// Compute the projection matrix
|
|
Mtx_PerspStereoTilt(&projection, 40.0f*M_PI/180.0f, 400.0f/240.0f, 0.01f, 1000.0f, iod, 2.0f);
|
|
|
|
// Calculate the modelView matrix
|
|
C3D_Mtx modelView;
|
|
Mtx_Identity(&modelView);
|
|
Mtx_Translate(&modelView, 0.0, 0.0, -4.0 + sinf(angleX));
|
|
Mtx_RotateX(&modelView, angleX, true);
|
|
Mtx_RotateY(&modelView, angleY, true);
|
|
|
|
// Rotate the cube each frame
|
|
if (iod >= 0.0f)
|
|
{
|
|
angleX += M_PI / 180;
|
|
angleY += M_PI / 360;
|
|
}
|
|
|
|
// Update the uniforms
|
|
C3D_FVUnifMtx(GPU_VERTEX_SHADER, uLoc_projection, &projection);
|
|
C3D_FVUnifMtx(GPU_VERTEX_SHADER, uLoc_modelView, &modelView);
|
|
|
|
// Draw the VBO
|
|
C3D_DrawArrays(GPU_TRIANGLES, 0, vertex_list_count);
|
|
}
|
|
|
|
static void sceneExit(void)
|
|
{
|
|
// Free the VBO
|
|
linearFree(vbo_data);
|
|
|
|
// Free the shader program
|
|
shaderProgramFree(&program);
|
|
DVLB_Free(vshader_dvlb);
|
|
}
|
|
|
|
#define CONFIG_3D_SLIDERSTATE (*(volatile float*)0x1FF81080)
|
|
|
|
int main()
|
|
{
|
|
// Initialize graphics
|
|
gfxInitDefault();
|
|
gfxSet3D(true);
|
|
|
|
C3D_Init(C3D_DEFAULT_CMDBUF_SIZE);
|
|
|
|
// Initialize the renderbuffer
|
|
static C3D_RenderBuf rb;
|
|
C3D_RenderBufInit(&rb, 240, 400, GPU_RB_RGBA8, GPU_RB_DEPTH24_STENCIL8);
|
|
rb.clearColor = CLEAR_COLOR;
|
|
C3D_RenderBufClear(&rb);
|
|
C3D_RenderBufBind(&rb);
|
|
|
|
// Initialize the scene
|
|
sceneInit();
|
|
|
|
// Main loop
|
|
while (aptMainLoop())
|
|
{
|
|
gspWaitForVBlank(); // Synchronize with the start of VBlank
|
|
gfxSwapBuffersGpu(); // Swap the framebuffers so that the frame that we rendered last frame is now visible
|
|
hidScanInput(); // Read the user input
|
|
|
|
// Respond to user input
|
|
u32 kDown = hidKeysDown();
|
|
if (kDown & KEY_START)
|
|
break; // break in order to return to hbmenu
|
|
|
|
float slider = CONFIG_3D_SLIDERSTATE;
|
|
float iod = slider/3;
|
|
|
|
// Render the scene
|
|
sceneRender(-iod);
|
|
C3D_Flush();
|
|
C3D_RenderBufTransfer(&rb, (u32*)gfxGetFramebuffer(GFX_TOP, GFX_LEFT, NULL, NULL), DISPLAY_TRANSFER_FLAGS);
|
|
C3D_RenderBufClear(&rb);
|
|
|
|
if (iod > 0.0f)
|
|
{
|
|
sceneRender(iod);
|
|
C3D_Flush();
|
|
C3D_RenderBufTransfer(&rb, (u32*)gfxGetFramebuffer(GFX_TOP, GFX_RIGHT, NULL, NULL), DISPLAY_TRANSFER_FLAGS);
|
|
C3D_RenderBufClear(&rb);
|
|
}
|
|
|
|
// Flush the framebuffers out of the data cache (not necessary with pure GPU rendering)
|
|
//gfxFlushBuffers();
|
|
}
|
|
|
|
// Deinitialize the scene
|
|
sceneExit();
|
|
|
|
// Deinitialize graphics
|
|
C3D_Fini();
|
|
gfxExit();
|
|
return 0;
|
|
}
|