#include #include #include typedef StateType state_update_t(void*); const char *StateTypeStr[] = { "null", #define State(name) #name, #include #undef State "quit", }; // Setup API for states #define State(name) \ typedef struct name##_state name##_state; \ typedef void (state_##name##_init_t)(name##_state*); \ typedef void (state_##name##_free_t)(name##_state*); \ typedef StateType (state_##name##_update_t)(name##_state*); #include #undef State #ifdef DAW_BUILD_HOTRELOAD // When hotreloading is enabled, we want to assign state function pointers // dynamically. #define State(name) \ state_##name##_init_t *name##_init = NULL; \ state_##name##_free_t *name##_free = NULL; \ state_##name##_update_t *name##_update = NULL; \ \ void* libstate_##name = NULL; \ const char* libstate_##name##_str = "lib" #name ".so"; #else // Otherwise we just declare them. #define State(name) \ extern state_##name##_init_t name##_init; \ extern state_##name##_free_t name##_free; \ extern state_##name##_update_t name##_update; #endif #include #undef State #include void State_init(StateType type, memory *mem) { switch (type) { #define State(name) \ case (STATE_##name): { \ name##_init(memory_allocate(mem, sizeof(name##_state))); \ break; \ } #include #undef State case STATE_null: case STATE_quit: DEBUG("Got %s state.\n", StateTypeStr[type]); break; default: exit(EXIT_FAILURE); } } void State_free(StateType type, memory *mem) { switch (type) { #define State(name) \ case (STATE_##name): { \ name##_free(mem->data); \ break; \ } #include #undef State case STATE_null: case STATE_quit: DEBUG("Got %s state.\n", StateTypeStr[type]); break; default: exit(EXIT_FAILURE); } memory_clear(mem); } StateType (*State_updateFunc(StateType type))(void*) { switch (type) { #ifdef DAW_BUILD_HOTRELOAD #define State(name) \ case (STATE_##name): { \ return (state_update_t*)name##_update; \ break; \ } #else #define State(name) \ case (STATE_##name): { \ return (state_update_t*)&name##_update; \ break; \ } #endif #include #undef State case STATE_null: case STATE_quit: return NULL; //DEBUG("Got %s state.\n", StateTypeStr[type]); break; default: exit(EXIT_FAILURE); } return NULL; } StateType State_update(StateType type, memory *mem) { StateType next_state = STATE_null; switch (type) { #define State(name) \ case (STATE_##name): { \ next_state = name##_update(mem->data); \ break; \ } #include #undef State case STATE_null: case STATE_quit: DEBUG("Got %s state.\n", StateTypeStr[type]); break; default: exit(EXIT_FAILURE); } return next_state; } #ifdef DAW_BUILD_HOTRELOAD bool State_reload(StateType type) { switch (type) { #define State(name) \ case (STATE_##name): { \ if (libstate_##name == NULL) { \ libstate_##name = dynamic_library_open(libstate_##name##_str); \ } else { \ libstate_##name = \ dynamic_library_reload(libstate_##name, libstate_##name##_str); \ } \ if (libstate_##name == NULL) { \ ERROR("Failed loading shared object: %s (%s)", \ libstate_##name##_str, \ dynamic_library_get_error()); \ return false; \ } \ \ name##_init = (state_##name##_init_t*)dynamic_library_get_symbol(libstate_##name, STR( name##_init ) ); \ name##_free = (state_##name##_free_t*)dynamic_library_get_symbol(libstate_##name, STR( name##_free ) ); \ name##_update = (state_##name##_update_t*)dynamic_library_get_symbol(libstate_##name, STR( name##_update ) ); \ break; \ } #include #undef State case STATE_null: case STATE_quit: ERROR("Invalid state"); DEBUG("Got %s state.\n", StateTypeStr[type]); break; default: exit(EXIT_FAILURE); } return true; } #endif