#include "engine/util/std.h" #include "engine/util/parser.h" #include "engine/image/image.h" #include "engine/raytracing.h" #include "engine/world.h" #include #include "engine/util/debug.h" #include #include //#include //#include #ifdef COMPILE_GUI #include #include #include #endif /** * Brett Terpstra 6920201 * */ using namespace Raytracing; extern bool* haltExecution; extern bool* pauseRaytracing; extern bool* haltRaytracing; int main(int argc, char** args) { // since this is linux only we can easily set our process priority to be high with a syscall // requires root. TODO: find way to doing this without root even if asking for user privilege escalation //setpriority(PRIO_PROCESS, 0, -20); // not a feature full parser but it'll work for what I need. Raytracing::Parser parser; parser.addOption("--single", "Enable Single Thread\n\tUse a single thread for ray tracing\n", "true"); // not implemented yet parser.addOption("--multi", "Enable Multi-threading\n" "\tUse multiple threads for ray tracing,\n" "\tYou can set the max threads using -t or --threads\n"); parser.addOption({{"-t"}, {"--threads"}}, "Max Usable Threads\n" "\tSet the max threads the ray tracer will attempt to use.\n" "\tDefaults to all cores of your cpu.\n", "0"); // not implemented yet parser.addOption({{"--gui"}, {"-g"}}, "Enable GUI\n" "\tWill create a GUI using X11 and display the image there.\n" "\tRequires the you compile with the option -DCOMPILE_GUI=ON. Will do nothing otherwise\n"); // not implemented yet parser.addOption({{"--gpu"}, {"-c"}}, "Enables GPU Compute\n" "\tRequires the --gui/-g flag enabled,\n" "\tWill use OpenGL compute shaders to render the image\n"); parser.addOption("--output", "Output Directory\n" "\tSet the output directory for the rendered image. Defaults to the local directory.\n", "./"); parser.addOption("--format", "Output Format\n" "\tSets the output format to BMP, PNG, or JPEG. \n", "PNG"); parser.addOption("-w", "Image Width\n" "\tSets the width of the output image.\n", "1440"); parser.addOption("-h", "Image Height\n" "\tSets the height of the output image.\n", "720"); parser.addOption("--fov", "Camera FOV\n" "\tSets the FOV used to render the camera.\n", "90"); parser.addOption("--resources", "Resources Directory\n" "\tSets the directory where the resources are stored.\n" "\tThis can be relative.Must have trailing '/' \n", "../resources/"); // disabled because don't currently have a way to parse vectors. TODO //parser.addOption("--position", "Camera Position\n\tSets the position used to render the scene with the camera.\n", "{0, 0, 0}"); // if the parser returns non-zero then it wants us to stop execution // likely due to a help function being called. if (parser.parse(args, argc)) return 0; // yes this is a very stupid and bad way of doing this. haltExecution = new bool; pauseRaytracing = new bool; haltRaytracing = new bool; *haltExecution = false; *pauseRaytracing = false; *haltRaytracing = false; if (signal(SIGTERM, [] (int sig) -> void { ilog<<"Computations complete.\nHalting now...\n"; *haltExecution = true; })==SIG_ERR) { elog<<"Unable to change signal handler.\n"; return 1; } if (signal(SIGINT, [] (int sig) -> void { ilog<<"Computations complete.\nHalting now...\n"; *haltExecution = true; })==SIG_ERR) { elog<<"Unable to change signal handler.\n"; return 1; } tlog << "Parsing complete! Starting raytracer with options:" << std::endl; // not perfect (contains duplicates) but good enough. parser.printAllInInfo(); Raytracing::Image image(1440, 720); //Raytracing::Image image(std::stoi(parser.getOptionValue("-w")), std::stoi(parser.getOptionValue("-h"))); Raytracing::Camera camera(std::stoi(parser.getOptionValue("--fov")), image); //camera.setPosition({0, 0, 1}); camera.setPosition({6, 5, 6}); camera.lookAt({0, 0, 0}); Raytracing::World world; Raytracing::OBJLoader loader; // assumes you are running it from a subdirectory, "build" or "cmake-build-release", etc. Raytracing::ModelData spider = loader.loadModel(parser.getOptionValue("--resources") + "spider.obj"); Raytracing::ModelData house = loader.loadModel(parser.getOptionValue("--resources") + "house.obj"); Raytracing::ModelData plane = loader.loadModel(parser.getOptionValue("--resources") + "plane.obj"); world.addMaterial("greenDiffuse", new Raytracing::DiffuseMaterial{Raytracing::Vec4{0, 1.0, 0, 1}}); world.addMaterial("redDiffuse", new Raytracing::DiffuseMaterial{Raytracing::Vec4{1.0, 0, 0, 1}}); world.addMaterial("blueDiffuse", new Raytracing::DiffuseMaterial{Raytracing::Vec4{0, 0, 1.0, 1}}); world.addMaterial("greenMetal", new Raytracing::MetalMaterial{Raytracing::Vec4{0.4, 1.0, 0.4, 1}}); world.addMaterial("redMetal", new Raytracing::BrushedMetalMaterial{Raytracing::Vec4{1.0, 0.4, 0.4, 1}, 0.6f}); world.addMaterial("blueMetal", new Raytracing::MetalMaterial{Raytracing::Vec4{0.4, 0.4, 1.0, 1}}); //world.add(new Raytracing::SphereObject({0,0,-1,0{}, 0.5, world.getMaterial("redDiffuse"))); //world.add(new Raytracing::SphereObject({-1,0,-1,0}, 0.5, world.getMaterial("blueMetal"))); //world.add(new Raytracing::SphereObject({1,0,-1,0}, 0.5, world.getMaterial("redMetal"))); world.add(new Raytracing::SphereObject({0, -100.5, -1, 0}, 100, world.getMaterial("greenDiffuse"))); //world.add(new Raytracing::TriangleObject({0,0.1,-0.5f,0}, {{-0.5, -0.5, 0.0}, {0.5, -0.5, 0.0}, {0.0, 0.5, 0}}, world.getMaterial("greenDiffuse"))); world.add(new Raytracing::ModelObject({0, 1, 0}, spider, world.getMaterial("redDiffuse"))); world.add(new Raytracing::ModelObject({-5, 0.5, 0}, plane, world.getMaterial("greenMetal"))); world.add(new Raytracing::ModelObject({5, 1, 0}, house, world.getMaterial("redDiffuse"))); world.add(new Raytracing::ModelObject({0, 0, -5}, house, world.getMaterial("blueDiffuse"))); world.add(new Raytracing::ModelObject({0, 0, 5}, house, world.getMaterial("blueDiffuse"))); if (parser.hasOption("--gui") || parser.hasOption("-g")) { #ifdef COMPILE_GUI XWindow window(1440, 720); Raytracing::Raycaster raycaster {camera, image, world, parser}; Texture mainImage(&image); auto spiderVAO = new VAO(spider.toTriangles()); auto houseVAO = new VAO(house.toTriangles()); auto planeVAO = new VAO(plane.toTriangles()); Shader shader("../resources/shaders/basic.vs", "../resources/shaders/basic.fs"); Shader worldShader("../resources/shaders/world.vs", "../resources/shaders/world.fs"); Raytracing::DisplayRenderer renderer {window, mainImage, shader, worldShader, raycaster, parser, spiderVAO, houseVAO, planeVAO, camera}; while (!window.shouldWindowClose()) { window.beginUpdate(); renderer.draw(); glPolygonMode(GL_FRONT_AND_BACK, GL_LINE); world.drawBVH(worldShader); glPolygonMode(GL_FRONT_AND_BACK, GL_FILL); window.endUpdate(); } *haltExecution = true; raycaster.join(); delete(spiderVAO); delete(houseVAO); delete(planeVAO); #else flog << "Program not compiled with GUI support! Unable to continue!\n"; #endif } else { Raytracing::Raycaster raycaster {camera, image, world, parser}; // run the raycaster the standard way ilog << "Running raycaster!\n"; if(parser.hasOption("--multi")) { raycaster.runMulti(std::max(std::stoi(parser.getOptionValue("-t")), std::stoi(parser.getOptionValue("--threads")))); } else { // we don't actually have to check for --single since it's implied to be default true. raycaster.runSingle(); } raycaster.join(); } profiler::print("Raytracer Results"); Raytracing::ImageOutput imageOutput(image); auto t = std::time(nullptr); auto now = std::localtime(&t); std::stringstream timeString; timeString << (1900 + now->tm_year); timeString << "-"; timeString << (1 + now->tm_mon); timeString << "-"; timeString << now->tm_mday; timeString << " "; timeString << now->tm_hour; timeString << ":"; timeString << now->tm_min; timeString << ":"; timeString << now->tm_sec; ilog << "Writing Image!\n"; imageOutput.write(parser.getOptionValue("--output") + timeString.str(), parser.getOptionValue("--format")); delete(haltExecution); delete(haltRaytracing); delete(pauseRaytracing); #ifdef COMPILE_GUI deleteQuad(); #endif return 0; }