silly code

main
Brett 2025-04-02 21:19:22 -04:00
parent 6747974360
commit 0cb0bdc672
7 changed files with 171 additions and 15 deletions

View File

@ -1,5 +1,5 @@
cmake_minimum_required(VERSION 3.25)
project(image-gp-2 VERSION 0.0.4)
project(image-gp-2 VERSION 0.0.5)
option(ENABLE_ADDRSAN "Enable the address sanitizer" OFF)
option(ENABLE_UBSAN "Enable the ub sanitizer" OFF)

View File

@ -18,6 +18,7 @@
#ifndef GP_SYSTEM_H
#define GP_SYSTEM_H
#include <image_storage.h>
#include <blt/std/types.h>
void setup_gp_system(blt::size_t population_size);
@ -26,4 +27,6 @@ void run_step();
bool should_terminate();
image_storage_t& get_image(blt::size_t index);
#endif //GP_SYSTEM_H

View File

@ -22,6 +22,7 @@
#include <array>
#include <atomic>
#include <string>
#include <blt/logging/logging.h>
#include <blt/std/types.h>
using image_pixel_t = float;
@ -49,12 +50,28 @@ struct image_storage_t
}
};
inline std::atomic_uint64_t g_allocated_nodes = 0;
inline std::atomic_uint64_t g_deallocated_nodes = 0;
struct atomic_node_t
{
explicit atomic_node_t(image_storage_t* data): data(data)
{
++g_allocated_nodes;
}
std::atomic<atomic_node_t*> next = nullptr;
image_storage_t* data = nullptr;
~atomic_node_t()
{
++g_deallocated_nodes;
}
};
inline std::atomic_uint64_t g_allocated_blocks = 0;
inline std::atomic_uint64_t g_deallocated_blocks = 0;
class atomic_list_t
{
public:
@ -106,14 +123,15 @@ struct image_t
delete front;
} else
data = new image_storage_t;
++g_allocated_blocks;
}
void drop()
{
const auto node = new atomic_node_t(); // NOLINT
node->data = data;
const auto node = new atomic_node_t(data); // NOLINT
data = nullptr;
g_image_list.push_back(node);
++g_deallocated_blocks;
}
[[nodiscard]] void* as_void_const() const

@ -1 +1 @@
Subproject commit abb4cc26a4ab2f739e18bdc6dd52bbce24a1b0d5
Subproject commit a742164bedd31cac6d828032ee1a4a18c64d0cac

View File

@ -30,11 +30,13 @@ gp_program program{
}
};
std::vector<image_storage_t> images;
image_storage_t reference_image = image_storage_t::from_file("../silly.png");
void fitness_func(const tree_t& tree, fitness_t& fitness, const blt::size_t)
void fitness_func(const tree_t& tree, fitness_t& fitness, const blt::size_t index)
{
auto image = tree.get_evaluation_ref<image_t>();
std::memcpy(images[index].data.data(), image->get_data().data.data(), IMAGE_SIZE_BYTES);
auto& data = image->get_data();
for (blt::size_t x = 0; x < IMAGE_DIMENSIONS; ++x)
{
@ -47,13 +49,14 @@ void fitness_func(const tree_t& tree, fitness_t& fitness, const blt::size_t)
const auto diff_r = data.get(x, y, 0) - reference_image.get(x, y, 0);
const auto diff_g = data.get(x, y, 1) - reference_image.get(x, y, 1);
const auto diff_b = data.get(x, y, 2) - reference_image.get(x, y, 2);
fitness.raw_fitness += diff_r * diff_r * multiplier + diff_g * diff_g * multiplier + diff_b * diff_b * multiplier;
const auto total = (diff_r + diff_g + diff_b) / 3;
fitness.raw_fitness += (total * total) * multiplier;
}
}
}
fitness.raw_fitness /= static_cast<float>(IMAGE_DIMENSIONS * IMAGE_DIMENSIONS);
fitness.raw_fitness /= static_cast<float>(IMAGE_SIZE_CHANNELS);
fitness.standardized_fitness = fitness.raw_fitness;
fitness.adjusted_fitness = fitness.standardized_fitness;
fitness.adjusted_fitness = -fitness.standardized_fitness;
}
void setup_operations()
@ -108,7 +111,11 @@ void setup_operations()
return std::exp(a);
}, "exp_float");
static operation_t op_log([](const float a) {
return a <= 0.0f ? 0.0f : std::log(a);
if (blt::f_equal(a, 0))
return 0.0f;
if (a < 0)
return -std::log(-a);
return std::log(a);
}, "log_float");
static auto lit = operation_t([]() {
return program.get_random().get_float(-1.0f, 1.0f);
@ -124,7 +131,9 @@ void setup_gp_system(const blt::size_t population_size)
{
prog_config_t config{};
config.population_size = population_size;
config.elites = 2;
program.set_config(config);
images.resize(population_size);
setup_operations();
static auto sel = select_tournament_t{};
program.generate_initial_population(program.get_typesystem().get_type<image_t>().id());
@ -151,3 +160,8 @@ bool should_terminate()
{
return program.should_terminate();
}
image_storage_t& get_image(blt::size_t index)
{
return images[index];
}

View File

@ -16,8 +16,6 @@
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
#include <image_storage.h>
#define STB_IMAGE_IMPLEMENTATION
#define STB_IMAGE_RESIZE_IMPLEMENTATION
#include <stb_image.h>
#include <stb_image_resize2.h>
#include <blt/iterator/zip.h>
@ -32,7 +30,7 @@ image_storage_t image_storage_t::from_file(const std::string& path)
image_storage_t storage{};
stbir_resize_float_linear(data, x, y, 0, storage.data.data(), IMAGE_DIMENSIONS, IMAGE_DIMENSIONS, 0, STBIR_RGBA);
STBI_FREE(data);
stbi_image_free(data);
return storage;
}

View File

@ -5,18 +5,49 @@
#include "blt/gfx/renderer/batch_2d_renderer.h"
#include "blt/gfx/renderer/camera.h"
#include <imgui.h>
#include <thread>
blt::gfx::matrix_state_manager global_matrices;
blt::gfx::resource_manager resources;
blt::gfx::batch_renderer_2d renderer_2d(resources, global_matrices);
blt::gfx::first_person_camera camera;
std::vector<blt::gfx::texture_gl2D*> gl_images;
blt::size_t population_size = 64;
namespace im = ImGui;
std::atomic_bool run_generation = false;
std::atomic_bool should_exit = false;
std::thread run_gp()
{
return std::thread{
[]() {
setup_gp_system(population_size);
while (!should_terminate() && !should_exit)
{
if (run_generation)
{
run_step();
run_generation = false;
} else
std::this_thread::sleep_for(std::chrono::milliseconds(10));
}
}
};
}
void init(const blt::gfx::window_data&)
{
using namespace blt::gfx;
setup_gp_system(64);
for (blt::size_t i = 0; i < population_size; i++)
{
auto texture = new texture_gl2D(IMAGE_DIMENSIONS, IMAGE_DIMENSIONS, GL_RGBA8);
gl_images.push_back(texture);
resources.set(std::to_string(i), texture);
}
global_matrices.create_internals();
resources.load_resources();
renderer_2d.create();
@ -30,11 +61,100 @@ void update(const blt::gfx::window_data& data)
camera.update_view(global_matrices);
global_matrices.update();
im::ShowDemoWindow();
ImGui::SetNextWindowPos(ImVec2(0, 0));
ImGui::SetNextWindowSize(ImVec2(ImGui::GetIO().DisplaySize.x, ImGui::GetIO().DisplaySize.y));
ImGui::Begin("MainWindow", nullptr,
ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoCollapse |
ImGuiWindowFlags_NoBringToFrontOnFocus);
// Create the tab bar
if (ImGui::BeginTabBar("MainTabs"))
{
// 1. Run GP tab
if (ImGui::BeginTabItem("Run GP"))
{
// Left child - fixed width (250px)
ImGui::BeginChild("ControlPanel", ImVec2(250, 0), true);
{
ImGui::Text("Control Panel");
ImGui::Separator();
if (ImGui::Button("Run Step"))
{
BLT_TRACE("Running step");
run_generation = true;
}
}
ImGui::EndChild();
// Right child - take the remaining space
ImGui::SameLine();
// ImGui::BeginChild("MainContent", ImVec2(0, 0), false, ImGuiWindowFlags_NoBackground);
{}
// ImGui::EndChild();
ImGui::EndTabItem();
}
// 2. Explore Trees tab
if (ImGui::BeginTabItem("Explore Trees"))
{
ImGui::Text("Here you can explore trees.");
// Additional UI for exploring trees
ImGui::EndTabItem();
}
// 3. Statistics tab
if (ImGui::BeginTabItem("Statistics"))
{
ImGui::Text("Here you can view statistics.");
// Additional UI for statistical data
ImGui::EndTabItem();
}
ImGui::EndTabBar();
}
ImGui::End(); // MainWindow
if (ImGui::Begin("Debug"))
{
const auto allocated_nodes = g_allocated_nodes.load(std::memory_order_relaxed);
const auto deallocated_nodes = g_deallocated_nodes.load(std::memory_order_relaxed);
const auto allocated_blocks = g_allocated_blocks.load(std::memory_order_relaxed);
const auto deallocated_blocks = g_deallocated_blocks.load(std::memory_order_relaxed);
ImGui::Text("Allocated Nodes / Deallocated Nodes: (%ld / %ld) (%lf%% / %ld)", allocated_nodes, deallocated_nodes,
(static_cast<double>(deallocated_nodes) / static_cast<double>(allocated_nodes)) * 100.0, allocated_nodes - deallocated_nodes);
ImGui::Text("Allocated Blocks / Deallocated Blocks: (%ld / %ld) (%lf%% / %ld)", allocated_blocks, deallocated_blocks,
(static_cast<double>(deallocated_blocks) / static_cast<double>(allocated_blocks)) * 100.0, allocated_blocks - deallocated_blocks);
}
ImGui::End();
for (blt::size_t i = 0; i < population_size; i++)
gl_images[i]->upload(get_image(i).data.data(), IMAGE_DIMENSIONS, IMAGE_DIMENSIONS, GL_RGBA, GL_FLOAT);
for (int i = 0; i < 10; i++)
{
for (int j = 0; j < 6; j++)
{
constexpr float padding_x = 32;
constexpr float padding_y = 32;
constexpr float img_width = 128;
constexpr float img_height = 128;
const float x = 256 + static_cast<float>(i) * img_width + padding_x * static_cast<float>(i) + img_width;
const float y = static_cast<float>(data.height) - (16 + static_cast<float>(j) * img_height + padding_y * static_cast<float>(j) +
img_height);
renderer_2d.drawRectangle(blt::gfx::rectangle2d_t{x, y, img_width, img_height}, std::to_string(i * j));
}
}
renderer_2d.render(data.width, data.height);
}
void destroy(const blt::gfx::window_data&)
{
gl_images.clear();
global_matrices.cleanup();
resources.cleanup();
renderer_2d.cleanup();
@ -43,5 +163,8 @@ void destroy(const blt::gfx::window_data&)
int main()
{
auto run_gp_thread = run_gp();
blt::gfx::init(blt::gfx::window_data{"Image GP", init, update, destroy}.setSyncInterval(1));
}
should_exit = true;
run_gp_thread.join();
}