#include "imgui.h" #include "imgui_impl_glfw.h" #include "imgui_impl_opengl3.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define STB_IMAGE_RESIZE_IMPLEMENTATION #include #include #include #ifdef _WIN32 #include #endif #ifndef INSTALL_PREFIX #ifdef __APPLE__ #define INSTALL_PREFIX "/usr/local" #else #define INSTALL_PREFIX "/usr" #endif #endif const char* OPENGL_VERSIONS_GLSL[] = { "#version 120", "#version 300 es", "#version 120" }; const int OPENGL_VERSIONS_MAJOR[] = { 3, 3, 2 }; const int OPENGL_VERSIONS_MINOR[] = { 0, 1, 1 }; const bool OPENGL_VERSIONS_IS_ES[] = { false, true, false }; #define OPENGL_VERSION_COUNT (sizeof(OPENGL_VERSIONS_GLSL) / sizeof(char*)) namespace core { ConfigManager configManager; ModuleManager moduleManager; ModuleComManager modComManager; GLFWwindow* window; void setInputSampleRate(double samplerate) { sigpath::signalPath.sourceSampleRate = samplerate; double effectiveSr = samplerate / ((double)(1 << sigpath::signalPath.decimation)); // NOTE: Zoom controls won't work spdlog::info("New DSP samplerate: {0} (source samplerate is {1})", effectiveSr, samplerate); gui::waterfall.setBandwidth(effectiveSr); gui::waterfall.setViewOffset(0); gui::waterfall.setViewBandwidth(effectiveSr); sigpath::signalPath.setSampleRate(effectiveSr); gui::mainWindow.setViewBandwidthSlider(effectiveSr); } }; bool maximized = false; bool fullScreen = false; static void glfw_error_callback(int error, const char* description) { spdlog::error("Glfw Error {0}: {1}", error, description); } static void maximized_callback(GLFWwindow* window, int n) { if (n == GLFW_TRUE) { maximized = true; } else { maximized = false; } } // main int sdrpp_main(int argc, char *argv[]) { spdlog::info("SDR++ v" VERSION_STR); // Load default options and parse command line options::loadDefaults(); if (!options::parse(argc, argv)) { return -1; } #ifdef _WIN32 if (!options::opts.showConsole) { FreeConsole(); } #endif // Check root directory if (!std::filesystem::exists(options::opts.root)) { spdlog::warn("Root directory {0} does not exist, creating it", options::opts.root); if (!std::filesystem::create_directory(options::opts.root)) { spdlog::error("Could not create root directory {0}", options::opts.root); return -1; } } if (!std::filesystem::is_directory(options::opts.root)) { spdlog::error("{0} is not a directory", options::opts.root); return -1; } // ======== DEFAULT CONFIG ======== json defConfig; defConfig["bandColors"]["amateur"] = "#FF0000FF"; defConfig["bandColors"]["aviation"] = "#00FF00FF"; defConfig["bandColors"]["broadcast"] = "#0000FFFF"; defConfig["bandColors"]["marine"] = "#00FFFFFF"; defConfig["bandColors"]["military"] = "#FFFF00FF"; defConfig["bandPlan"] = "General"; defConfig["bandPlanEnabled"] = true; defConfig["bandPlanPos"] = 0; defConfig["centerTuning"] = false; defConfig["colorMap"] = "Classic"; defConfig["fastFFT"] = false; defConfig["fftHeight"] = 300; defConfig["fftRate"] = 20; defConfig["fftSize"] = 65536; defConfig["fftWindow"] = 1; defConfig["frequency"] = 100000000.0; defConfig["fullWaterfallUpdate"] = false; defConfig["max"] = 0.0; defConfig["maximized"] = false; // Menu defConfig["menuElements"] = json::array(); defConfig["menuElements"][0]["name"] = "Source"; defConfig["menuElements"][0]["open"] = true; defConfig["menuElements"][1]["name"] = "Radio"; defConfig["menuElements"][1]["open"] = true; defConfig["menuElements"][2]["name"] = "Recorder"; defConfig["menuElements"][2]["open"] = true; defConfig["menuElements"][3]["name"] = "Sinks"; defConfig["menuElements"][3]["open"] = true; defConfig["menuElements"][3]["name"] = "Frequency Manager"; defConfig["menuElements"][3]["open"] = true; defConfig["menuElements"][4]["name"] = "VFO Color"; defConfig["menuElements"][4]["open"] = true; defConfig["menuElements"][5]["name"] = "Scripting"; defConfig["menuElements"][5]["open"] = false; defConfig["menuElements"][6]["name"] = "Band Plan"; defConfig["menuElements"][6]["open"] = true; defConfig["menuElements"][7]["name"] = "Display"; defConfig["menuElements"][7]["open"] = true; defConfig["menuWidth"] = 300; defConfig["min"] = -120.0; // Module instances defConfig["moduleInstances"]["Airspy Source"]["module"] = "airspy_source"; defConfig["moduleInstances"]["Airspy Source"]["enabled"] = true; defConfig["moduleInstances"]["AirspyHF+ Source"]["module"] = "airspyhf_source"; defConfig["moduleInstances"]["AirspyHF+ Source"]["enabled"] = true; defConfig["moduleInstances"]["BladeRF Source"]["module"] = "bladerf_source"; defConfig["moduleInstances"]["BladeRF Source"]["enabled"] = true; defConfig["moduleInstances"]["File Source"]["module"] = "file_source"; defConfig["moduleInstances"]["File Source"]["enabled"] = true; defConfig["moduleInstances"]["HackRF Source"]["module"] = "hackrf_source"; defConfig["moduleInstances"]["HackRF Source"]["enabled"] = true; defConfig["moduleInstances"]["LimeSDR Source"]["module"] = "limesdr_source"; defConfig["moduleInstances"]["LimeSDR Source"]["enabled"] = true; defConfig["moduleInstances"]["RTL-SDR Source"]["module"] = "rtl_sdr_source"; defConfig["moduleInstances"]["RTL-SDR Source"]["enabled"] = true; defConfig["moduleInstances"]["RTL-TCP Source"]["module"] = "rtl_tcp_source"; defConfig["moduleInstances"]["RTL-TCP Source"]["enabled"] = true; defConfig["moduleInstances"]["SDRplay Source"]["module"] = "sdrplay_source"; defConfig["moduleInstances"]["SDRplay Source"]["enabled"] = true; defConfig["moduleInstances"]["SoapySDR Source"]["module"] = "soapy_source"; defConfig["moduleInstances"]["SoapySDR Source"]["enabled"] = true; defConfig["moduleInstances"]["SpyServer Source"]["module"] = "spyserver_source"; defConfig["moduleInstances"]["SpyServer Source"]["enabled"] = true; defConfig["moduleInstances"]["PlutoSDR Source"]["module"] = "plutosdr_source"; defConfig["moduleInstances"]["PlutoSDR Source"]["enabled"] = true; defConfig["moduleInstances"]["Audio Sink"] = "audio_sink"; defConfig["moduleInstances"]["Radio"] = "radio"; defConfig["moduleInstances"]["Frequency Manager"] = "frequency_manager"; defConfig["moduleInstances"]["Recorder"] = "recorder"; defConfig["moduleInstances"]["Rigctl Server"] = "rigctl_server"; // Themes defConfig["theme"] = "Dark"; defConfig["modules"] = json::array(); defConfig["offsetMode"] = (int)0; // Off defConfig["offset"] = 0.0; defConfig["showMenu"] = true; defConfig["showWaterfall"] = true; defConfig["source"] = ""; defConfig["decimationPower"] = 0; defConfig["iqCorrection"] = false; defConfig["streams"]["Radio"]["muted"] = false; defConfig["streams"]["Radio"]["sink"] = "Audio"; defConfig["streams"]["Radio"]["volume"] = 1.0f; defConfig["windowSize"]["h"] = 720; defConfig["windowSize"]["w"] = 1280; defConfig["vfoOffsets"] = json::object(); defConfig["vfoColors"]["Radio"] = "#FFFFFF"; #ifdef _WIN32 defConfig["modulesDirectory"] = "./modules"; defConfig["resourcesDirectory"] = "./res"; #else defConfig["modulesDirectory"] = INSTALL_PREFIX "/lib/sdrpp/plugins"; defConfig["resourcesDirectory"] = INSTALL_PREFIX "/share/sdrpp"; #endif // Load config spdlog::info("Loading config"); core::configManager.setPath(options::opts.root + "/config.json"); core::configManager.load(defConfig); core::configManager.enableAutoSave(); core::configManager.acquire(); // Fix missing elements in config for (auto const& item : defConfig.items()) { if (!core::configManager.conf.contains(item.key())) { spdlog::info("Missing key in config {0}, repairing", item.key()); core::configManager.conf[item.key()] = defConfig[item.key()]; } } // Remove unused elements auto items = core::configManager.conf.items(); for (auto const& item : items) { if (!defConfig.contains(item.key())) { spdlog::info("Unused key in config {0}, repairing", item.key()); core::configManager.conf.erase(item.key()); } } // Update to new module representation in config if needed for (auto [_name, inst] : core::configManager.conf["moduleInstances"].items()) { if (!inst.is_string()) { continue; } std::string mod = inst; json newMod; newMod["module"] = mod; newMod["enabled"] = true; core::configManager.conf["moduleInstances"][_name] = newMod; } core::configManager.release(true); if (options::opts.serverMode) { return server_main(); } core::configManager.acquire(); int winWidth = core::configManager.conf["windowSize"]["w"]; int winHeight = core::configManager.conf["windowSize"]["h"]; maximized = core::configManager.conf["maximized"]; std::string resDir = core::configManager.conf["resourcesDirectory"]; json bandColors = core::configManager.conf["bandColors"]; core::configManager.release(); if (!std::filesystem::is_directory(resDir)) { spdlog::error("Resource directory doesn't exist! Please make sure that you've configured it correctly in config.json (check readme for details)"); return 1; } // Setup window glfwSetErrorCallback(glfw_error_callback); if (!glfwInit()) { return 1; } #ifdef __APPLE__ // GL 3.2 + GLSL 150 const char* glsl_version = "#version 150"; glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3); glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 2); glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE); // 3.2+ only glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE); // Required on Mac // Create window with graphics context GLFWmonitor* monitor = glfwGetPrimaryMonitor(); core::window = glfwCreateWindow(winWidth, winHeight, "SDR++ v" VERSION_STR " (Built at " __TIME__ ", " __DATE__ ")", NULL, NULL); if (core::window == NULL) return 1; glfwMakeContextCurrent(core::window); #else const char* glsl_version = "#version 120"; GLFWmonitor* monitor = NULL; for (int i = 0; i < OPENGL_VERSION_COUNT; i++) { glsl_version = OPENGL_VERSIONS_GLSL[i]; if (OPENGL_VERSIONS_IS_ES[i]) { glfwWindowHint(GLFW_CLIENT_API, GLFW_OPENGL_ES_API); } glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, OPENGL_VERSIONS_MAJOR[i]); glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, OPENGL_VERSIONS_MINOR[i]); // Create window with graphics context monitor = glfwGetPrimaryMonitor(); core::window = glfwCreateWindow(winWidth, winHeight, "SDR++ v" VERSION_STR " (Built at " __TIME__ ", " __DATE__ ")", NULL, NULL); if (core::window == NULL) { spdlog::info("OpenGL {0}.{1} {2}was not supported", OPENGL_VERSIONS_MAJOR[i], OPENGL_VERSIONS_MINOR[i], OPENGL_VERSIONS_IS_ES[i] ? "ES ": ""); continue; } spdlog::info("Using OpenGL {0}.{1}{2}", OPENGL_VERSIONS_MAJOR[i], OPENGL_VERSIONS_MINOR[i], OPENGL_VERSIONS_IS_ES[i] ? " ES": ""); glfwMakeContextCurrent(core::window); break; } #endif // Add callback for max/min if GLFW supports it #if (GLFW_VERSION_MAJOR == 3) && (GLFW_VERSION_MINOR >= 3) if (maximized) { glfwMaximizeWindow(core::window); } glfwSetWindowMaximizeCallback(core::window, maximized_callback); #endif // Load app icon if (!std::filesystem::is_regular_file(resDir + "/icons/sdrpp.png")) { spdlog::error("Icon file '{0}' doesn't exist!", resDir + "/icons/sdrpp.png"); return 1; } GLFWimage icons[10]; icons[0].pixels = stbi_load(((std::string)(resDir + "/icons/sdrpp.png")).c_str(), &icons[0].width, &icons[0].height, 0, 4); icons[1].pixels = (unsigned char*)malloc(16 * 16 * 4); icons[1].width = icons[1].height = 16; icons[2].pixels = (unsigned char*)malloc(24 * 24 * 4); icons[2].width = icons[2].height = 24; icons[3].pixels = (unsigned char*)malloc(32 * 32 * 4); icons[3].width = icons[3].height = 32; icons[4].pixels = (unsigned char*)malloc(48 * 48 * 4); icons[4].width = icons[4].height = 48; icons[5].pixels = (unsigned char*)malloc(64 * 64 * 4); icons[5].width = icons[5].height = 64; icons[6].pixels = (unsigned char*)malloc(96 * 96 * 4); icons[6].width = icons[6].height = 96; icons[7].pixels = (unsigned char*)malloc(128 * 128 * 4); icons[7].width = icons[7].height = 128; icons[8].pixels = (unsigned char*)malloc(196 * 196 * 4); icons[8].width = icons[8].height = 196; icons[9].pixels = (unsigned char*)malloc(256 * 256 * 4); icons[9].width = icons[9].height = 256; stbir_resize_uint8(icons[0].pixels, icons[0].width, icons[0].height, icons[0].width * 4, icons[1].pixels, 16, 16, 16 * 4, 4); stbir_resize_uint8(icons[0].pixels, icons[0].width, icons[0].height, icons[0].width * 4, icons[2].pixels, 24, 24, 24 * 4, 4); stbir_resize_uint8(icons[0].pixels, icons[0].width, icons[0].height, icons[0].width * 4, icons[3].pixels, 32, 32, 32 * 4, 4); stbir_resize_uint8(icons[0].pixels, icons[0].width, icons[0].height, icons[0].width * 4, icons[4].pixels, 48, 48, 48 * 4, 4); stbir_resize_uint8(icons[0].pixels, icons[0].width, icons[0].height, icons[0].width * 4, icons[5].pixels, 64, 64, 64 * 4, 4); stbir_resize_uint8(icons[0].pixels, icons[0].width, icons[0].height, icons[0].width * 4, icons[6].pixels, 96, 96, 96 * 4, 4); stbir_resize_uint8(icons[0].pixels, icons[0].width, icons[0].height, icons[0].width * 4, icons[7].pixels, 128, 128, 128 * 4, 4); stbir_resize_uint8(icons[0].pixels, icons[0].width, icons[0].height, icons[0].width * 4, icons[8].pixels, 196, 196, 196 * 4, 4); stbir_resize_uint8(icons[0].pixels, icons[0].width, icons[0].height, icons[0].width * 4, icons[9].pixels, 256, 256, 256 * 4, 4); glfwSetWindowIcon(core::window, 10, icons); stbi_image_free(icons[0].pixels); for (int i = 1; i < 10; i++) { free(icons[i].pixels); } if (glewInit() != GLEW_OK) { spdlog::error("Failed to initialize OpenGL loader!"); return 1; } // Setup Dear ImGui context IMGUI_CHECKVERSION(); ImGui::CreateContext(); ImGuiIO& io = ImGui::GetIO(); (void)io; io.IniFilename = NULL; // Setup Platform/Renderer bindings ImGui_ImplGlfw_InitForOpenGL(core::window, true); if (!ImGui_ImplOpenGL3_Init(glsl_version)) { // If init fail, try to fall back on GLSL 1.2 spdlog::warn("Could not init using OpenGL with normal GLSL version, falling back to GLSL 1.2"); if (!ImGui_ImplOpenGL3_Init("#version 120")) { spdlog::error("Failed to initialize OpenGL with GLSL 1.2"); return -1; } } if (!style::loadFonts(resDir)) { return -1; } thememenu::init(resDir); LoadingScreen::setWindow(core::window); LoadingScreen::show("Loading icons"); spdlog::info("Loading icons"); if (!icons::load(resDir)) { return -1; } LoadingScreen::show("Loading band plans"); spdlog::info("Loading band plans"); bandplan::loadFromDir(resDir + "/bandplans"); LoadingScreen::show("Loading band plan colors"); spdlog::info("Loading band plans color table"); bandplan::loadColorTable(bandColors); gui::mainWindow.init(); spdlog::info("Ready."); bool _maximized = maximized; int fsWidth, fsHeight, fsPosX, fsPosY; // Main loop while (!glfwWindowShouldClose(core::window)) { glfwPollEvents(); // Start the Dear ImGui frame ImGui_ImplOpenGL3_NewFrame(); ImGui_ImplGlfw_NewFrame(); ImGui::NewFrame(); //ImGui::ShowDemoWindow(); if (_maximized != maximized) { _maximized = maximized; core::configManager.acquire(); core::configManager.conf["maximized"]= _maximized; if (!maximized) { glfwSetWindowSize(core::window, core::configManager.conf["windowSize"]["w"], core::configManager.conf["windowSize"]["h"]); } core::configManager.release(true); } int _winWidth, _winHeight; glfwGetWindowSize(core::window, &_winWidth, &_winHeight); if (ImGui::IsKeyPressed(GLFW_KEY_F11)) { fullScreen = !fullScreen; if (fullScreen) { spdlog::info("Fullscreen: ON"); fsWidth = _winWidth; fsHeight = _winHeight; glfwGetWindowPos(core::window, &fsPosX, &fsPosY); const GLFWvidmode * mode = glfwGetVideoMode(glfwGetPrimaryMonitor()); glfwSetWindowMonitor(core::window, monitor, 0, 0, mode->width, mode->height, 0); } else { spdlog::info("Fullscreen: OFF"); glfwSetWindowMonitor(core::window, nullptr, fsPosX, fsPosY, fsWidth, fsHeight, 0); } } if ((_winWidth != winWidth || _winHeight != winHeight) && !maximized && _winWidth > 0 && _winHeight > 0) { winWidth = _winWidth; winHeight = _winHeight; core::configManager.acquire(); core::configManager.conf["windowSize"]["w"] = winWidth; core::configManager.conf["windowSize"]["h"] = winHeight; core::configManager.release(true); } if (winWidth > 0 && winHeight > 0) { ImGui::SetNextWindowPos(ImVec2(0, 0)); ImGui::SetNextWindowSize(ImVec2(_winWidth, _winHeight)); gui::mainWindow.draw(); } // Rendering ImGui::Render(); int display_w, display_h; glfwGetFramebufferSize(core::window, &display_w, &display_h); glViewport(0, 0, display_w, display_h); //glClearColor(0.0666f, 0.0666f, 0.0666f, 1.0f); glClearColor(gui::themeManager.clearColor.x, gui::themeManager.clearColor.y, gui::themeManager.clearColor.z, gui::themeManager.clearColor.w); glClear(GL_COLOR_BUFFER_BIT); ImGui_ImplOpenGL3_RenderDrawData(ImGui::GetDrawData()); glfwSwapInterval(1); // Enable vsync glfwSwapBuffers(core::window); } // Shut down all modules for (auto& [name, mod] : core::moduleManager.modules) { mod.end(); } // Cleanup ImGui_ImplOpenGL3_Shutdown(); ImGui_ImplGlfw_Shutdown(); ImGui::DestroyContext(); glfwDestroyWindow(core::window); glfwTerminate(); sigpath::signalPath.stop(); core::configManager.disableAutoSave(); core::configManager.save(); return 0; }