#include <3ds.h> #include #include #include "vshader_shbin.h" constexpr u32 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); constexpr u32 FRAMEBUFFER_TRANSFER_FLAGS = GX_TRANSFER_RAW_COPY(1); struct vertex { float st[2]; float coords[3]; float uv[2]; }; static DVLB_s* vshaderDVLB; static shaderProgram_s program; static s8 spheresUniformLocation; static s8 sphereColorsUniformLocation; static s8 sphereLightsUniformLocation; static s8 randUniformLocation; static s8 shiftUniformLocation; static constexpr unsigned int VERTEX_COUNT_W = 120; static constexpr unsigned int VERTEX_COUNT_H = 10 * VERTEX_COUNT_W / 6; static constexpr unsigned int VERTEX_COUNT = VERTEX_COUNT_W * VERTEX_COUNT_H; static vertex vertexList[VERTEX_COUNT]; static u16 vertexIndices[VERTEX_COUNT * 2 + VERTEX_COUNT_H * 2]; static void* vboData; static void* iboData; static C3D_Tex prevFrameLeft; static C3D_Tex prevFrameRight; class camera { public: camera(const C3D_FVec lookfrom, const C3D_FVec lookat, const C3D_FVec vup, const float vfov, const float aspectRatio, const float iod) { const float theta = C3D_AngleFromDegrees(vfov); const float h = tanf(theta / 2.0f); const float viewportHeight = 2.0f * h; const float viewportWidth = aspectRatio * viewportHeight; const float screen = FVec3_Distance(lookfrom, lookat) / 2.0f; shift = iod / (-400.0f * screen * viewportWidth); const C3D_FVec w = FVec3_Normalize(FVec3_Subtract(lookfrom, lookat)); const C3D_FVec u = FVec3_Normalize(FVec3_Cross(vup, w)); const C3D_FVec v = FVec3_Cross(w, u); origin = FVec3_Add(lookfrom, FVec3_New(-iod / 3.0f, 0, 0)); horizontal = FVec3_Scale(u, viewportWidth); vertical = FVec3_Scale(v, viewportHeight); lowerLeftCorner = FVec3_Add( FVec3_Subtract(FVec3_Add(FVec3_Scale(horizontal, -0.5f), FVec3_Scale(vertical, -0.5f)), w), FVec3_New(-iod / 3.0f, 0, 0) ); } void setupFixed() const { *C3D_FixedAttribGetWritePtr(0) = origin; *C3D_FixedAttribGetWritePtr(1) = lowerLeftCorner; *C3D_FixedAttribGetWritePtr(2) = horizontal; *C3D_FixedAttribGetWritePtr(3) = vertical; } void setupUniform() const { C3D_FVUnifSet(GPU_VERTEX_SHADER, shiftUniformLocation, shift / 200.0f, 240.0f / 256.0f, 400.0f / 512.0f, 0); } private: C3D_FVec origin; C3D_FVec lowerLeftCorner; C3D_FVec horizontal; C3D_FVec vertical; float shift; }; static void setupCam(C3D_FVec lookFrom, C3D_FVec lookAt, C3D_FVec up, float iod) { constexpr float VFOV = 90; constexpr float ASPECT_RATIO = C3D_AspectRatioTop; camera cam(lookFrom, lookAt, up, VFOV, ASPECT_RATIO, iod); cam.setupFixed(); cam.setupUniform(); } static float rand01() { return static_cast(rand()) / RAND_MAX; } static C3D_FVec randvec() { C3D_FVec out; out.w = rand01(); do { out.x = rand01() * 2 - 1; out.y = rand01() * 2 - 1; out.z = rand01() * 2 - 1; } while (FVec3_Magnitude(out) > 1.0f); return out; return FVec4_New(0, 0, 0, 0); } static void setupVertices() { for (unsigned int x = 0; x < VERTEX_COUNT_W; x++) { for (unsigned int y = 0; y < VERTEX_COUNT_H; y++) { vertex& v = vertexList[x + y * VERTEX_COUNT_W]; const float s = static_cast(x) / VERTEX_COUNT_W; const float t = static_cast(y) / VERTEX_COUNT_H; // swapped due to 3DS screen orientation v.st[0] = t + rand01() / VERTEX_COUNT_W; v.st[1] = s + rand01() / VERTEX_COUNT_H; v.coords[0] = 2.0f * s - 1.0f; v.coords[1] = 2.0f * t - 1.0f; v.coords[2] = -0.05f; v.uv[0] = s * (240.0f / 256.0f); v.uv[1] = t * (400.0f / 512.0f); } } memcpy(vboData, vertexList, sizeof(vertexList)); } static void setupRandom() { constexpr int NUM_RAND = 10; C3D_FVec* randPtr = C3D_FVUnifWritePtr(GPU_VERTEX_SHADER, randUniformLocation, NUM_RAND); for (int i = 0; i < NUM_RAND; i++) { randPtr[i] = randvec(); } } static void sceneInit() { vboData = linearAlloc(sizeof(vertexList)); unsigned int v = 0; for (unsigned int y = 0; y < VERTEX_COUNT_H; y++) { vertexIndices[v++] = y * VERTEX_COUNT_W; for (unsigned int x = 0; x < VERTEX_COUNT_W; x++) { vertexIndices[v++] = y * VERTEX_COUNT_W + x; vertexIndices[v++] = (y + 1) * VERTEX_COUNT_W + x; } vertexIndices[v++] = (y + 2) * VERTEX_COUNT_W - 1; } vshaderDVLB = DVLB_ParseFile((u32*)vshader_shbin, vshader_shbin_size); shaderProgramInit(&program); shaderProgramSetVsh(&program, &vshaderDVLB->DVLE[0]); C3D_BindProgram(&program); spheresUniformLocation = shaderInstanceGetUniformLocation(program.vertexShader, "spheres"); sphereColorsUniformLocation = shaderInstanceGetUniformLocation(program.vertexShader, "sphereColors"); sphereLightsUniformLocation = shaderInstanceGetUniformLocation(program.vertexShader, "sphereLights"); randUniformLocation = shaderInstanceGetUniformLocation(program.vertexShader, "rand"); shiftUniformLocation = shaderInstanceGetUniformLocation(program.vertexShader, "shift"); C3D_AttrInfo* attrInfo = C3D_GetAttrInfo(); AttrInfo_Init(attrInfo); AttrInfo_AddFixed(attrInfo, 0); AttrInfo_AddFixed(attrInfo, 1); AttrInfo_AddFixed(attrInfo, 2); AttrInfo_AddFixed(attrInfo, 3); AttrInfo_AddLoader(attrInfo, 4, GPU_FLOAT, 2); AttrInfo_AddLoader(attrInfo, 5, GPU_FLOAT, 3); AttrInfo_AddLoader(attrInfo, 6, GPU_FLOAT, 2); iboData = linearAlloc(sizeof(vertexIndices)); memcpy(iboData, vertexIndices, sizeof(vertexIndices)); C3D_BufInfo* bufInfo = C3D_GetBufInfo(); BufInfo_Init(bufInfo); BufInfo_Add(bufInfo, vboData, sizeof(vertex), 3, 0x654); C3D_TexEnv* env = C3D_GetTexEnv(0); C3D_TexEnvInit(env); C3D_TexEnvSrc(env, C3D_Both, GPU_PRIMARY_COLOR, GPU_TEXTURE0, GPU_CONSTANT); C3D_TexEnvFunc(env, C3D_RGB, GPU_INTERPOLATE); C3D_TexEnvFunc(env, C3D_Alpha, GPU_REPLACE); constexpr u16 NUM_SPHERES = 4; C3D_FVec* spheres = C3D_FVUnifWritePtr(GPU_VERTEX_SHADER, spheresUniformLocation, NUM_SPHERES); spheres[0] = FVec4_New(0.0f, -100.5f, -1.0f, 100.0f); spheres[1] = FVec4_New(0.6f, 0.0f, -1.0f, 0.5f); spheres[2] = FVec4_New(-0.6f, 0.0f, -1.0f, 0.5f); spheres[3] = FVec4_New(0.0f, 10.0f, -1.0f, 9.0f); C3D_FVec* sphereColors = C3D_FVUnifWritePtr(GPU_VERTEX_SHADER, sphereColorsUniformLocation, NUM_SPHERES); sphereColors[0] = FVec3_New(0.9f, 0.3f, 0.3f); sphereColors[1] = FVec3_New(0.3f, 0.9f, 0.3f); sphereColors[2] = FVec3_New(0.3f, 0.3f, 0.9f); sphereColors[3] = FVec3_New(1.0f, 1.0f, 1.0f); C3D_FVec* sphereLights = C3D_FVUnifWritePtr(GPU_VERTEX_SHADER, sphereLightsUniformLocation, NUM_SPHERES); sphereLights[0] = FVec3_New(0.0f, 0.0f, 0.0f); sphereLights[1] = FVec3_New(0.0f, 0.0f, 0.0f); sphereLights[2] = FVec3_New(0.0f, 0.0f, 0.0f); sphereLights[3] = FVec3_New(1.0f, 1.0f, 1.0f); } static void sceneRender() { C3D_DrawElements(GPU_TRIANGLE_STRIP, VERTEX_COUNT * 2 + VERTEX_COUNT_H * 2, C3D_UNSIGNED_SHORT, iboData); } static void sceneExit() { linearFree(vboData); linearFree(iboData); shaderProgramFree(&program); DVLB_Free(vshaderDVLB); } static float clamp(float t, float min, float max) { return t < min ? min : (t > max ? max : t); } static u32 shift(u8 value, u32 amt) { return static_cast(value) << amt; } static u32 getFrameNumColor(unsigned int frameNum) { if (frameNum == 0) { return 0xFFFFFFFF; } else { const float ratio = 1 - static_cast(frameNum) / (frameNum + 1); const float scaledRatio = clamp(255.0f * ratio, 0.0f, 255.0f); const u8 bits = static_cast(scaledRatio); return shift(bits, 24) | shift(bits, 16) | shift(bits, 8) | bits; } } int main(int argc, char* argv[]) { gfxInitDefault(); gfxSet3D(true); C3D_Init(C3D_DEFAULT_CMDBUF_SIZE); C3D_RenderTarget* targetLeft = C3D_RenderTargetCreate(240, 400, GPU_RB_RGBA8, GPU_RB_DEPTH24_STENCIL8); C3D_RenderTargetSetOutput(targetLeft, GFX_TOP, GFX_LEFT, DISPLAY_TRANSFER_FLAGS); C3D_RenderTarget* targetRight = C3D_RenderTargetCreate(240, 400, GPU_RB_RGBA8, GPU_RB_DEPTH24_STENCIL8); C3D_RenderTargetSetOutput(targetRight, GFX_TOP, GFX_RIGHT, DISPLAY_TRANSFER_FLAGS); C3D_TexInit(&prevFrameLeft, 256, 512, GPU_RGBA8); C3D_TexInit(&prevFrameRight, 256, 512, GPU_RGBA8); sceneInit(); C3D_FVec lookFrom = FVec3_New(0, 0.6125f, 0); C3D_FVec lookAt = FVec3_New(0, 0, -1.0f); C3D_FVec vup = FVec3_New(0, 1.0f, 0); bool dirty = true; unsigned int frameNum = 0; float previousIOD = osGet3DSliderState() / 4.0f; // Main loop while (aptMainLoop()) { hidScanInput(); const u32 kDown = hidKeysHeld(); if (kDown & KEY_START) { break; // break in order to return to hbmenu } constexpr float MOVE_RATE = 1.0f / 30.0f; C3D_FVec diff = FVec3_New(0, 0, 0); if (kDown & KEY_DLEFT) { diff.x -= MOVE_RATE; dirty = true; } if (kDown & KEY_DRIGHT) { diff.x += MOVE_RATE; dirty = true; } if (kDown & KEY_DUP) { diff.z += MOVE_RATE; dirty = true; } if (kDown & KEY_DDOWN) { diff.z -= MOVE_RATE; dirty = true; } if (kDown & KEY_L) { diff.y -= MOVE_RATE; dirty = true; } if (kDown & KEY_R) { diff.y += MOVE_RATE; dirty = true; } lookFrom = FVec3_Add(lookFrom, diff); const float iod = osGet3DSliderState() / 4.0f; if (fabsf(iod - previousIOD) > 0.01f) { dirty = true; } if (dirty) { frameNum = 0; previousIOD = iod; dirty = false; } setupVertices(); setupRandom(); C3D_TexEnvColor(C3D_GetTexEnv(0), getFrameNumColor(frameNum)); if (frameNum < 256 && C3D_FrameBegin(C3D_FRAME_SYNCDRAW)) { setupCam(lookFrom, lookAt, vup, -iod); C3D_TexBind(0, &prevFrameLeft); C3D_RenderTargetClear(targetLeft, C3D_CLEAR_ALL, 0, 0); C3D_FrameDrawOn(targetLeft); sceneRender(); if (iod > 0) { setupCam(lookFrom, lookAt, vup, iod); C3D_TexBind(0, &prevFrameRight); C3D_RenderTargetClear(targetRight, C3D_CLEAR_ALL, 0, 0); C3D_FrameDrawOn(targetRight); sceneRender(); } C3D_FrameEnd(0); C3D_SyncTextureCopy( (u32*)targetLeft->frameBuf.colorBuf, GX_BUFFER_DIM((240 * 8 * 4) >> 4, 0), (u32*)prevFrameLeft.data + (512 - 400) * 256, GX_BUFFER_DIM((240 * 8 * 4) >> 4, ((256 - 240) * 8 * 4) >> 4), 240 * 400 * 4, FRAMEBUFFER_TRANSFER_FLAGS ); if (iod > 0) { C3D_SyncTextureCopy( (u32*)targetRight->frameBuf.colorBuf, GX_BUFFER_DIM((240 * 8 * 4) >> 4, 0), (u32*)prevFrameRight.data + (512 - 400) * 256, GX_BUFFER_DIM((240 * 8 * 4) >> 4, ((256 - 240) * 8 * 4) >> 4), 240 * 400 * 4, FRAMEBUFFER_TRANSFER_FLAGS ); } frameNum++; } } sceneExit(); C3D_TexDelete(&prevFrameLeft); C3D_RenderTargetDelete(targetLeft); C3D_Fini(); gfxExit(); return 0; }