2024-03-03 07:43:07 +01:00
|
|
|
#define SDL_MAIN_USE_CALLBACKS 1
|
2024-03-03 10:25:14 +01:00
|
|
|
#include "SDL3/SDL_blendmode.h"
|
2024-12-16 06:09:00 +01:00
|
|
|
#include "SDL3/SDL_camera.h"
|
2024-03-03 10:25:14 +01:00
|
|
|
#include "SDL3/SDL_error.h"
|
2024-12-16 06:09:00 +01:00
|
|
|
#include "SDL3/SDL_events.h"
|
2024-03-03 07:43:07 +01:00
|
|
|
#include "SDL3/SDL_init.h"
|
2024-03-03 11:39:46 +01:00
|
|
|
#include "SDL3/SDL_keycode.h"
|
2024-03-03 07:43:07 +01:00
|
|
|
#include "SDL3/SDL_log.h"
|
|
|
|
#include "SDL3/SDL_main.h"
|
|
|
|
#include "SDL3/SDL_pixels.h"
|
|
|
|
#include "SDL3/SDL_render.h"
|
2024-03-03 10:25:14 +01:00
|
|
|
#include "SDL3/SDL_stdinc.h"
|
|
|
|
#include "SDL3/SDL_surface.h"
|
2024-03-03 07:43:07 +01:00
|
|
|
#include "SDL3/SDL_timer.h"
|
|
|
|
#include "SDL3/SDL_video.h"
|
|
|
|
|
2024-03-03 10:25:14 +01:00
|
|
|
#include "mgba/core/config.h"
|
2024-03-03 07:43:07 +01:00
|
|
|
#include "mgba/core/core.h"
|
|
|
|
#include "mgba/core/interface.h"
|
|
|
|
#include "mgba/feature/commandline.h"
|
2024-12-16 06:09:00 +01:00
|
|
|
|
|
|
|
#include <stdio.h>
|
2024-12-16 09:55:03 +01:00
|
|
|
#include <string.h>
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
#define NUM_CHANNELS 3
|
2024-03-03 10:25:14 +01:00
|
|
|
#define FRAMESKIP_LIMIT 20
|
2024-03-03 09:52:02 +01:00
|
|
|
|
2024-03-03 10:41:29 +01:00
|
|
|
#if BYTES_PER_PIXEL == 4
|
|
|
|
#define SCREEN_FMT SDL_PIXELFORMAT_XBGR8888
|
|
|
|
#elif BYTES_PER_PIXEL == 2
|
|
|
|
#define SCREEN_FMT SDL_PIXELFORMAT_XBGR1555
|
|
|
|
#else
|
|
|
|
#error "unknown pixel format"
|
|
|
|
#endif
|
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
static SDL_Texture* textures[NUM_CHANNELS] = { NULL, NULL, NULL };
|
2024-12-16 03:36:56 +01:00
|
|
|
static bool texture_updated = false;
|
|
|
|
static SDL_CameraID front_camera = 0;
|
|
|
|
static SDL_CameraID back_camera = 0;
|
2024-03-03 07:43:07 +01:00
|
|
|
static SDL_Surface* scaled = NULL;
|
2024-03-03 09:52:02 +01:00
|
|
|
static SDL_Surface* filtered[NUM_CHANNELS] = { NULL, NULL, NULL };
|
|
|
|
static SDL_Surface* screens[NUM_CHANNELS] = { NULL, NULL, NULL };
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
static struct mCore* cores[NUM_CHANNELS] = { NULL, NULL, NULL };
|
2024-03-03 11:39:46 +01:00
|
|
|
static unsigned keys = 0;
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
struct AppState {
|
|
|
|
SDL_Camera* camera;
|
|
|
|
SDL_Window* window;
|
|
|
|
SDL_Renderer* renderer;
|
|
|
|
int cam_idx, spec_idx;
|
|
|
|
};
|
|
|
|
|
2024-03-03 07:43:07 +01:00
|
|
|
enum mColorFormat pixfmt_sdl_to_mgba(uint32_t sdl_fmt) {
|
|
|
|
switch (sdl_fmt) {
|
|
|
|
case SDL_PIXELFORMAT_ABGR1555: return mCOLOR_BGR5;
|
|
|
|
case SDL_PIXELFORMAT_ABGR8888: return mCOLOR_ABGR8;
|
|
|
|
case SDL_PIXELFORMAT_ARGB1555: return mCOLOR_RGB5;
|
|
|
|
case SDL_PIXELFORMAT_ARGB8888: return mCOLOR_ARGB8;
|
|
|
|
case SDL_PIXELFORMAT_BGR565: return mCOLOR_BGR565;
|
|
|
|
case SDL_PIXELFORMAT_BGRA8888: return mCOLOR_BGRA8;
|
|
|
|
case SDL_PIXELFORMAT_BGRX8888: return mCOLOR_BGRX8;
|
|
|
|
case SDL_PIXELFORMAT_RGB565: return mCOLOR_RGB565;
|
|
|
|
case SDL_PIXELFORMAT_RGBA8888: return mCOLOR_RGBA8;
|
|
|
|
case SDL_PIXELFORMAT_RGBX8888: return mCOLOR_RGBX8;
|
|
|
|
case SDL_PIXELFORMAT_XBGR1555: return mCOLOR_BGR5;
|
|
|
|
case SDL_PIXELFORMAT_XBGR8888: return mCOLOR_XBGR8;
|
|
|
|
case SDL_PIXELFORMAT_XRGB1555: return mCOLOR_RGB5;
|
|
|
|
case SDL_PIXELFORMAT_XRGB8888: return mCOLOR_XRGB8;
|
2024-03-03 10:41:29 +01:00
|
|
|
/* others don't quite match between mgba and SDL */
|
2024-03-03 07:43:07 +01:00
|
|
|
default: return mCOLOR_ANY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
void myStartRequestImageRed(struct mImageSource* self, unsigned w, unsigned h, int colorFormats) {
|
2024-03-03 07:43:07 +01:00
|
|
|
SDL_DestroySurface(scaled);
|
2024-03-03 09:52:02 +01:00
|
|
|
SDL_DestroySurface(filtered[0]);
|
2024-03-03 07:43:07 +01:00
|
|
|
scaled = SDL_CreateSurface(w, h, SDL_PIXELFORMAT_XBGR1555);
|
2024-03-03 09:52:02 +01:00
|
|
|
filtered[0] = SDL_CreateSurface(w, h, SDL_PIXELFORMAT_XBGR1555);
|
|
|
|
SDL_Log("red: %d x %d", w, h);
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
void myStartRequestImageGreen(struct mImageSource* self, unsigned w, unsigned h, int colorFormats) {
|
|
|
|
SDL_DestroySurface(filtered[1]);
|
|
|
|
filtered[1] = SDL_CreateSurface(w, h, SDL_PIXELFORMAT_XBGR1555);
|
|
|
|
SDL_Log("green: %d x %d", w, h);
|
|
|
|
}
|
|
|
|
|
|
|
|
void myStartRequestImageBlue(struct mImageSource* self, unsigned w, unsigned h, int colorFormats) {
|
|
|
|
SDL_DestroySurface(filtered[2]);
|
|
|
|
filtered[2] = SDL_CreateSurface(w, h, SDL_PIXELFORMAT_XBGR1555);
|
|
|
|
SDL_Log("blue: %d x %d", w, h);
|
|
|
|
}
|
|
|
|
|
|
|
|
void myStopRequestImageRed(struct mImageSource* self) {
|
2024-03-03 07:43:07 +01:00
|
|
|
SDL_DestroySurface(scaled);
|
2024-03-03 09:52:02 +01:00
|
|
|
SDL_DestroySurface(filtered[0]);
|
2024-03-03 07:43:07 +01:00
|
|
|
scaled = NULL;
|
2024-03-03 09:52:02 +01:00
|
|
|
filtered[0] = NULL;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
void myStopRequestImageGreen(struct mImageSource* self) {
|
|
|
|
SDL_DestroySurface(filtered[1]);
|
|
|
|
filtered[1] = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void myStopRequestImageBlue(struct mImageSource* self) {
|
|
|
|
SDL_DestroySurface(filtered[2]);
|
|
|
|
filtered[2] = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void myRequestImageRed(struct mImageSource* self, const void** buf, size_t* stride, enum mColorFormat* fmt) {
|
|
|
|
SDL_Surface* source = filtered[0];
|
|
|
|
if (source != NULL) {
|
2024-12-16 03:36:56 +01:00
|
|
|
*fmt = pixfmt_sdl_to_mgba(source->format);
|
|
|
|
*stride = source->pitch / (BYTES_PER_PIXEL / SDL_BYTESPERPIXEL(source->format));
|
2024-03-03 09:52:02 +01:00
|
|
|
*buf = source->pixels;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void myRequestImageGreen(struct mImageSource* self, const void** buf, size_t* stride, enum mColorFormat* fmt) {
|
|
|
|
SDL_Surface* source = filtered[1];
|
|
|
|
if (source != NULL) {
|
2024-12-16 03:36:56 +01:00
|
|
|
*fmt = pixfmt_sdl_to_mgba(source->format);
|
|
|
|
*stride = source->pitch / (BYTES_PER_PIXEL / SDL_BYTESPERPIXEL(source->format));
|
2024-03-03 09:52:02 +01:00
|
|
|
*buf = source->pixels;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
void myRequestImageBlue(struct mImageSource* self, const void** buf, size_t* stride, enum mColorFormat* fmt) {
|
|
|
|
SDL_Surface* source = filtered[2];
|
|
|
|
if (source != NULL) {
|
2024-12-16 03:36:56 +01:00
|
|
|
*fmt = pixfmt_sdl_to_mgba(source->format);
|
|
|
|
*stride = source->pitch / (BYTES_PER_PIXEL / SDL_BYTESPERPIXEL(source->format));
|
2024-03-03 09:52:02 +01:00
|
|
|
*buf = source->pixels;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-03 10:41:29 +01:00
|
|
|
struct mImageSource gb_img_src[3];
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
bool SelectCamera(struct AppState* st) {
|
2024-03-03 07:43:07 +01:00
|
|
|
int devcount = 0;
|
2024-12-16 09:55:03 +01:00
|
|
|
SDL_CameraID *devices = SDL_GetCameras(&devcount);
|
2024-12-16 06:09:00 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
SDL_Renderer* renderer = st->renderer;
|
2024-12-16 06:09:00 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
if (st->cam_idx >= devcount) {
|
|
|
|
st->cam_idx = 0;
|
|
|
|
} else if (st->cam_idx < 0) {
|
|
|
|
st->cam_idx = devcount - 1;
|
|
|
|
}
|
2024-12-16 06:09:00 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
if (!devices) {
|
|
|
|
SDL_Log("SDL_GetCameras failed: %s", SDL_GetError());
|
|
|
|
return false;
|
|
|
|
}
|
2024-12-16 06:09:00 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
if (devcount == 0) {
|
|
|
|
SDL_free(devices);
|
2024-12-16 06:09:00 +01:00
|
|
|
|
|
|
|
SDL_SetRenderDrawColor(renderer, 0, 0, 0, 0);
|
|
|
|
SDL_RenderClear(renderer);
|
|
|
|
|
|
|
|
SDL_SetRenderDrawColor(renderer, 255, 255, 255, 255);
|
2024-12-16 09:55:03 +01:00
|
|
|
SDL_RenderDebugText(renderer, 4, 4, "connect a camera");
|
2024-12-16 06:09:00 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
SDL_RenderPresent(renderer);
|
|
|
|
}
|
2024-12-16 06:09:00 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
const SDL_CameraID device = devices[st->cam_idx];
|
|
|
|
const char *name = SDL_GetCameraName(device);
|
2024-12-16 06:09:00 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
const SDL_CameraPosition position = SDL_GetCameraPosition(device);
|
|
|
|
const char *posstr = "";
|
2024-12-16 06:09:00 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
char snprintfbuf[20];
|
2024-12-16 06:09:00 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
int formats_len = 0;
|
|
|
|
|
|
|
|
SDL_CameraSpec** formats = SDL_GetCameraSupportedFormats(device, &formats_len);
|
|
|
|
if (st->spec_idx >= formats_len) {
|
|
|
|
st->spec_idx = 0;
|
|
|
|
} else if (st->spec_idx < 0) {
|
|
|
|
st->spec_idx = formats_len - 1;
|
|
|
|
}
|
|
|
|
SDL_CameraSpec* format = formats[st->spec_idx];
|
|
|
|
const char* pixfmt_name = SDL_GetPixelFormatName(format->format);
|
|
|
|
if (strncmp("SDL_PIXELFORMAT_", pixfmt_name, 16) == 0) {
|
|
|
|
pixfmt_name += 16;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (position == SDL_CAMERA_POSITION_FRONT_FACING) {
|
|
|
|
front_camera = device;
|
|
|
|
posstr = "[frontfacing]";
|
|
|
|
} else if (position == SDL_CAMERA_POSITION_BACK_FACING) {
|
|
|
|
back_camera = device;
|
|
|
|
posstr = "[backfacing]";
|
2024-12-16 06:09:00 +01:00
|
|
|
}
|
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
SDL_SetRenderDrawColor(renderer, 0, 0, 0, 0);
|
|
|
|
SDL_RenderClear(renderer);
|
|
|
|
|
|
|
|
SDL_SetRenderDrawColor(renderer, 255, 255, 255, 255);
|
|
|
|
SDL_RenderDebugText(renderer, 4, 4, "camera: (up/down)");
|
|
|
|
snprintf(snprintfbuf, sizeof(snprintfbuf), "%d. %s", st->cam_idx, posstr);
|
|
|
|
SDL_RenderDebugText(renderer, 8, 16, snprintfbuf);
|
|
|
|
SDL_RenderDebugText(renderer, 4, 28, name);
|
|
|
|
SDL_free((void*)name);
|
|
|
|
|
|
|
|
SDL_RenderDebugText(renderer, 4, 52, "mode: (left/right)");
|
|
|
|
snprintf(
|
|
|
|
snprintfbuf,
|
|
|
|
sizeof(snprintfbuf),
|
|
|
|
"%d.",
|
|
|
|
st->spec_idx
|
|
|
|
);
|
|
|
|
SDL_RenderDebugText(renderer, 8, 64, snprintfbuf);
|
|
|
|
|
|
|
|
snprintf(
|
|
|
|
snprintfbuf,
|
|
|
|
sizeof(snprintfbuf),
|
|
|
|
"%dx%d",
|
|
|
|
format->width,
|
|
|
|
format->height
|
|
|
|
);
|
|
|
|
SDL_RenderDebugText(renderer, 4, 76, snprintfbuf);
|
|
|
|
|
|
|
|
snprintf(
|
|
|
|
snprintfbuf,
|
|
|
|
sizeof(snprintfbuf),
|
|
|
|
"%.1f Hz",
|
|
|
|
((float)format->framerate_numerator / (float)format->framerate_denominator)
|
|
|
|
);
|
|
|
|
SDL_RenderDebugText(renderer, 4, 88, snprintfbuf);
|
|
|
|
SDL_RenderDebugText(renderer, 4, 100, pixfmt_name);
|
|
|
|
|
|
|
|
SDL_RenderDebugText(renderer, 4, 124, "confirm: (enter)");
|
|
|
|
|
|
|
|
SDL_free(formats);
|
|
|
|
SDL_free(devices);
|
|
|
|
|
|
|
|
SDL_RenderPresent(renderer);
|
|
|
|
|
2024-12-16 06:09:00 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_AppResult SDL_AppInit(void** appstate, int argc, char *argv[]) {
|
2024-03-03 07:43:07 +01:00
|
|
|
int i;
|
2024-03-03 10:41:29 +01:00
|
|
|
unsigned w, h;
|
|
|
|
|
2024-03-03 10:02:01 +01:00
|
|
|
struct mArguments args;
|
2024-03-03 07:43:07 +01:00
|
|
|
bool parsed = mArgumentsParse(&args, argc, argv, NULL, 0);
|
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
struct AppState* st = malloc(sizeof(struct AppState));
|
|
|
|
memset(st, 0, sizeof(struct AppState));
|
|
|
|
*appstate = st;
|
|
|
|
|
2024-03-03 07:43:07 +01:00
|
|
|
/* Enable standard application logging */
|
2024-12-16 03:36:56 +01:00
|
|
|
SDL_SetLogPriority(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_INFO);
|
2024-03-03 07:43:07 +01:00
|
|
|
|
|
|
|
if (!parsed) {
|
|
|
|
SDL_Log("Couldn't parse arguments");
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_FAILURE;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-03-03 10:41:29 +01:00
|
|
|
gb_img_src[0].requestImage = myRequestImageRed;
|
|
|
|
gb_img_src[0].startRequestImage = myStartRequestImageRed;
|
|
|
|
gb_img_src[0].stopRequestImage = myStopRequestImageRed;
|
|
|
|
gb_img_src[1].requestImage = myRequestImageGreen;
|
|
|
|
gb_img_src[1].startRequestImage = myStartRequestImageGreen;
|
|
|
|
gb_img_src[1].stopRequestImage = myStopRequestImageGreen;
|
|
|
|
gb_img_src[2].requestImage = myRequestImageBlue;
|
|
|
|
gb_img_src[2].startRequestImage = myStartRequestImageBlue;
|
|
|
|
gb_img_src[2].stopRequestImage = myStopRequestImageBlue;
|
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
for (i = 0; i < NUM_CHANNELS; i++) {
|
|
|
|
struct mCore* core = mCoreFind(args.fname);
|
|
|
|
if (!core || !core->init(core)) {
|
|
|
|
SDL_Log("Couldn't initialize mgba core!");
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_FAILURE;
|
2024-03-03 09:52:02 +01:00
|
|
|
}
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
if (!mCoreLoadFile(core, args.fname)) {
|
|
|
|
SDL_Log("Failed to load ROM");
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_FAILURE;
|
2024-03-03 09:52:02 +01:00
|
|
|
}
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
mCoreConfigInit(&core->config, NULL);
|
2024-03-03 10:41:29 +01:00
|
|
|
/*mCoreConfigLoad(&core->config);*/
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
mArgumentsApply(&args, NULL, 0, &core->config);
|
|
|
|
mCoreConfigSetDefaultValue(&core->config, "idleOptimization", "detect");
|
2024-03-05 00:44:49 +01:00
|
|
|
mCoreConfigSetDefaultValue(&core->config, "frameskip", "4");
|
2024-03-03 09:52:02 +01:00
|
|
|
mCoreConfigSetDefaultValue(&core->config, "sgb.borders", "0");
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
mCoreLoadConfig(core);
|
|
|
|
|
|
|
|
cores[i] = core;
|
|
|
|
}
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 10:41:29 +01:00
|
|
|
/* without loss of generality */
|
2024-12-16 03:36:56 +01:00
|
|
|
cores[0]->desiredVideoDimensions(cores[0], &w, &h);
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-12-16 03:36:56 +01:00
|
|
|
if (!SDL_Init(SDL_INIT_VIDEO | SDL_INIT_CAMERA)) {
|
2024-03-03 07:43:07 +01:00
|
|
|
SDL_Log("Couldn't initialize SDL3: %s", SDL_GetError());
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_FAILURE;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
st->window = SDL_CreateWindow("cgbwebcam", w, h, SDL_WINDOW_RESIZABLE);
|
|
|
|
if (!st->window) {
|
2024-03-03 07:43:07 +01:00
|
|
|
SDL_Log("Couldn't create window: %s", SDL_GetError());
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_FAILURE;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
st->renderer = SDL_CreateRenderer(st->window, NULL);
|
|
|
|
if (!st->renderer) {
|
2024-03-03 07:43:07 +01:00
|
|
|
SDL_Log("Couldn't create renderer: %s", SDL_GetError());
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_FAILURE;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-12-16 03:36:56 +01:00
|
|
|
SDL_SetLogPriorities(SDL_LOG_PRIORITY_VERBOSE);
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
for (i = 0; i < NUM_CHANNELS; i++) {
|
|
|
|
int j;
|
|
|
|
struct mCore* core = cores[i];
|
2024-03-03 10:41:29 +01:00
|
|
|
SDL_Texture* texture;
|
|
|
|
SDL_Surface* screen = SDL_CreateSurface(w, h, SCREEN_FMT);
|
2024-03-03 09:52:02 +01:00
|
|
|
|
|
|
|
core->setVideoBuffer(core, screen->pixels, screen->pitch / BYTES_PER_PIXEL);
|
|
|
|
screens[i] = screen;
|
|
|
|
|
|
|
|
/* Create texture with appropriate format */
|
2024-12-16 09:55:03 +01:00
|
|
|
texture = SDL_CreateTextureFromSurface(st->renderer, screens[0]);
|
2024-03-03 09:52:02 +01:00
|
|
|
if (!texture) {
|
|
|
|
SDL_Log("Couldn't create texture: %s", SDL_GetError());
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_FAILURE;
|
2024-03-03 09:52:02 +01:00
|
|
|
}
|
|
|
|
SDL_SetTextureBlendMode(texture, SDL_BLENDMODE_ADD);
|
|
|
|
textures[i] = texture;
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 10:41:29 +01:00
|
|
|
core->setPeripheral(core, mPERIPH_IMAGE_SOURCE, &gb_img_src[i]);
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
core->reset(core);
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
core->setKeys(core, 0);
|
|
|
|
for (j = 0; j < 600; j++) {
|
|
|
|
core->runFrame(core);
|
|
|
|
}
|
2024-03-03 10:41:29 +01:00
|
|
|
core->setKeys(core, 1); /* title screen */
|
2024-03-03 07:43:07 +01:00
|
|
|
core->runFrame(core);
|
2024-03-03 09:52:02 +01:00
|
|
|
core->setKeys(core, 0);
|
|
|
|
for (j = 0; j < 25; j++) {
|
|
|
|
core->runFrame(core);
|
|
|
|
}
|
2024-03-03 10:41:29 +01:00
|
|
|
core->setKeys(core, 1); /* select 'shoot' */
|
2024-03-03 07:43:07 +01:00
|
|
|
core->runFrame(core);
|
2024-03-03 09:52:02 +01:00
|
|
|
core->setKeys(core, 0);
|
|
|
|
for (j = 0; j < 75; j++) {
|
|
|
|
core->runFrame(core);
|
|
|
|
}
|
2024-03-03 10:41:29 +01:00
|
|
|
core->setKeys(core, 1); /* select 'shoot' again */
|
2024-03-03 07:43:07 +01:00
|
|
|
core->runFrame(core);
|
2024-03-03 09:52:02 +01:00
|
|
|
core->setKeys(core, 0);
|
|
|
|
for (j = 0; j < 100; j++) {
|
|
|
|
core->runFrame(core);
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_Log("prepared core %d", i);
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
SDL_SetTextureColorMod(textures[0], 255, 0, 0);
|
|
|
|
SDL_SetTextureColorMod(textures[1], 0, 255, 0);
|
|
|
|
SDL_SetTextureColorMod(textures[2], 0, 0, 255);
|
|
|
|
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_CONTINUE; /* start the main app loop. */
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
static SDL_AppResult FlipCamera(struct AppState* st) {
|
2024-03-03 07:43:07 +01:00
|
|
|
static Uint64 last_flip = 0;
|
|
|
|
if ((SDL_GetTicks() - last_flip) < 3000) { /* must wait at least 3 seconds between flips. */
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_CONTINUE;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
if (st->camera) {
|
|
|
|
const SDL_CameraID current = SDL_GetCameraID(st->camera);
|
2024-12-16 03:36:56 +01:00
|
|
|
SDL_CameraID nextcam = 0;
|
2024-03-03 07:43:07 +01:00
|
|
|
if (current == front_camera) {
|
|
|
|
nextcam = back_camera;
|
|
|
|
} else if (current == back_camera) {
|
|
|
|
nextcam = front_camera;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nextcam) {
|
|
|
|
SDL_Log("Flip camera!");
|
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
SDL_CloseCamera(st->camera);
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
st->camera = SDL_OpenCamera(nextcam, NULL);
|
|
|
|
if (!st->camera) {
|
2024-03-03 07:43:07 +01:00
|
|
|
SDL_Log("Failed to open camera device: %s", SDL_GetError());
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_FAILURE;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
last_flip = SDL_GetTicks();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_CONTINUE;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-12-16 03:36:56 +01:00
|
|
|
SDL_AppResult SDL_AppEvent(void *appstate, SDL_Event *event) {
|
2024-12-16 09:55:03 +01:00
|
|
|
struct AppState* st = (struct AppState*)appstate;
|
|
|
|
|
|
|
|
if (st->camera == NULL) {
|
|
|
|
switch (event->type) {
|
|
|
|
case SDL_EVENT_QUIT:
|
|
|
|
SDL_Log("Quit!");
|
|
|
|
return false;
|
|
|
|
|
|
|
|
case SDL_EVENT_KEY_DOWN:
|
|
|
|
switch (event->key.key) {
|
|
|
|
case SDLK_UP: st->cam_idx -= 1; break;
|
|
|
|
case SDLK_DOWN: st->cam_idx += 1; break;
|
|
|
|
case SDLK_LEFT: st->spec_idx -= 1; break;
|
|
|
|
case SDLK_RIGHT: st->spec_idx += 1; break;
|
|
|
|
case SDLK_RETURN: {
|
|
|
|
SDL_CameraID *devices = SDL_GetCameras(NULL);
|
|
|
|
SDL_CameraID device = devices[st->cam_idx];
|
|
|
|
SDL_CameraSpec** formats = SDL_GetCameraSupportedFormats(device, NULL);
|
|
|
|
st->camera = SDL_OpenCamera(device, formats[st->spec_idx]);
|
|
|
|
SDL_free(devices);
|
|
|
|
SDL_free(formats);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-03 07:43:07 +01:00
|
|
|
switch (event->type) {
|
|
|
|
case SDL_EVENT_KEY_DOWN: {
|
2024-12-16 03:36:56 +01:00
|
|
|
const SDL_Keycode sym = event->key.key;
|
2024-03-03 11:39:46 +01:00
|
|
|
switch (sym) {
|
2024-12-16 03:36:56 +01:00
|
|
|
case SDLK_ESCAPE: case SDLK_AC_BACK: return SDL_APP_SUCCESS;
|
2024-12-16 09:55:03 +01:00
|
|
|
case SDLK_SPACE: FlipCamera(st); break;
|
2024-03-03 11:39:46 +01:00
|
|
|
case SDLK_UP: keys |= 0x40; break;
|
|
|
|
case SDLK_DOWN: keys |= 0x80; break;
|
|
|
|
case SDLK_LEFT: keys |= 0x20; break;
|
|
|
|
case SDLK_RIGHT: keys |= 0x10; break;
|
2024-12-16 03:36:56 +01:00
|
|
|
case SDLK_A: case SDLK_Z: keys |= 1; break;
|
|
|
|
case SDLK_B: case SDLK_X: keys |= 2; break;
|
2024-03-03 11:39:46 +01:00
|
|
|
case SDLK_RETURN: keys |= 8; break;
|
|
|
|
case SDLK_BACKSPACE: keys |= 4; break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case SDL_EVENT_KEY_UP: {
|
2024-12-16 03:36:56 +01:00
|
|
|
const SDL_Keycode sym = event->key.key;
|
2024-03-03 11:39:46 +01:00
|
|
|
switch (sym) {
|
|
|
|
case SDLK_UP: keys &= ~0x40; break;
|
|
|
|
case SDLK_DOWN: keys &= ~0x80; break;
|
|
|
|
case SDLK_LEFT: keys &= ~0x20; break;
|
|
|
|
case SDLK_RIGHT: keys &= ~0x10; break;
|
2024-12-16 03:36:56 +01:00
|
|
|
case SDLK_A: case SDLK_Z: keys &= ~1; break;
|
|
|
|
case SDLK_B: case SDLK_X: keys &= ~2; break;
|
2024-03-03 11:39:46 +01:00
|
|
|
case SDLK_RETURN: keys &= ~8; break;
|
|
|
|
case SDLK_BACKSPACE: keys &= ~4; break;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case SDL_EVENT_MOUSE_BUTTON_DOWN:
|
|
|
|
/* !!! FIXME: only flip if clicked in the area of a "flip" icon. */
|
2024-12-16 09:55:03 +01:00
|
|
|
return FlipCamera(st);
|
2024-03-03 07:43:07 +01:00
|
|
|
|
|
|
|
case SDL_EVENT_QUIT:
|
|
|
|
SDL_Log("Quit!");
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_SUCCESS;
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
case SDL_EVENT_CAMERA_DEVICE_APPROVED: {
|
|
|
|
SDL_CameraSpec spec;
|
2024-03-03 07:43:07 +01:00
|
|
|
SDL_Log("Camera approved!");
|
2024-12-16 09:55:03 +01:00
|
|
|
if (!SDL_GetCameraFormat(st->camera, &spec)) {
|
2024-03-03 07:43:07 +01:00
|
|
|
SDL_Log("Couldn't get camera spec: %s", SDL_GetError());
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_FAILURE;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
break;
|
2024-12-16 09:55:03 +01:00
|
|
|
}
|
2024-03-03 07:43:07 +01:00
|
|
|
|
|
|
|
case SDL_EVENT_CAMERA_DEVICE_DENIED:
|
|
|
|
SDL_Log("Camera denied!");
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_FAILURE;
|
2024-12-16 08:06:30 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
case SDL_EVENT_WINDOW_RESIZED: {
|
|
|
|
Sint32 winw = event->window.data1, winh = event->window.data2;
|
|
|
|
SDL_UpdateWindowSurface(window);
|
|
|
|
if (TODO: need to correct aspect ratio?) {
|
|
|
|
if (!SDL_SetWindowSize(window, winw/2, winh)) {
|
|
|
|
SDL_Log("Failed to resize window to %dx%d: %s", winw, winh, SDL_GetError());
|
|
|
|
return SDL_APP_FAILURE;
|
|
|
|
} else {
|
|
|
|
SDL_Log("Resized window to %dx%d", winw, winh);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_CONTINUE;
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-12-16 03:36:56 +01:00
|
|
|
SDL_AppResult SDL_AppIterate(void* appstate) {
|
2024-12-16 09:55:03 +01:00
|
|
|
struct AppState* st = (struct AppState*)appstate;
|
|
|
|
|
|
|
|
if (st->camera == NULL) {
|
|
|
|
if (!SelectCamera(st)) {
|
|
|
|
SDL_Log("Failed to show camera device menu: %s", SDL_GetError());
|
|
|
|
return SDL_APP_FAILURE;
|
|
|
|
}
|
|
|
|
return SDL_APP_CONTINUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_Renderer* renderer = st->renderer;
|
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
int i;
|
2024-03-03 10:25:14 +01:00
|
|
|
static int since_last_present = 0;
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
if (scaled != NULL) {
|
|
|
|
Uint64 timestampNS = 0;
|
2024-12-16 09:55:03 +01:00
|
|
|
SDL_Surface *frame = st->camera ? SDL_AcquireCameraFrame(st->camera, ×tampNS) : NULL;
|
2024-03-03 09:52:02 +01:00
|
|
|
|
|
|
|
if (frame) {
|
2024-12-16 03:36:56 +01:00
|
|
|
struct SDL_Rect srcrect, dstrect;
|
|
|
|
SDL_GetSurfaceClipRect(frame, &srcrect);
|
|
|
|
SDL_GetSurfaceClipRect(scaled, &dstrect);
|
|
|
|
int w = srcrect.h * dstrect.w / dstrect.h;
|
2024-03-03 10:41:29 +01:00
|
|
|
|
|
|
|
/*SDL_Log("new frame %d x %d %s", frame->w, frame->h, SDL_GetPixelFormatName(frame->format->format));*/
|
2024-12-16 03:36:56 +01:00
|
|
|
texture_updated = false;
|
2024-03-03 10:41:29 +01:00
|
|
|
|
2024-12-16 03:36:56 +01:00
|
|
|
if (w <= srcrect.w) {
|
2024-03-03 09:52:02 +01:00
|
|
|
srcrect.x = (srcrect.w - w) / 2;
|
|
|
|
srcrect.w = w;
|
|
|
|
} else {
|
2024-12-16 03:36:56 +01:00
|
|
|
int h = srcrect.w * dstrect.h / dstrect.w;
|
2024-03-03 09:52:02 +01:00
|
|
|
srcrect.y = (srcrect.h - h) / 2;
|
|
|
|
srcrect.h = h;
|
|
|
|
}
|
2024-12-16 03:36:56 +01:00
|
|
|
if (!SDL_BlitSurfaceScaled(frame, &srcrect, scaled, &dstrect, SDL_SCALEMODE_NEAREST)) {
|
2024-03-03 09:52:02 +01:00
|
|
|
SDL_Log("failed to scale: %s", SDL_GetError());
|
|
|
|
}
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
SDL_ReleaseCameraFrame(st->camera, frame);
|
2024-12-16 06:09:00 +01:00
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
SDL_SetSurfaceColorMod(scaled, 255, 0, 0);
|
|
|
|
SDL_BlitSurface(scaled, NULL, filtered[0], NULL);
|
|
|
|
SDL_SetSurfaceColorMod(scaled, 0, 255, 0);
|
|
|
|
SDL_BlitSurface(scaled, NULL, filtered[1], NULL);
|
|
|
|
SDL_SetSurfaceColorMod(scaled, 0, 0, 255);
|
|
|
|
SDL_BlitSurface(scaled, NULL, filtered[2], NULL);
|
|
|
|
}
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-03-03 10:25:14 +01:00
|
|
|
if (since_last_present < FRAMESKIP_LIMIT) {
|
|
|
|
for (i = 0; i < NUM_CHANNELS; i++) {
|
|
|
|
struct mCore* core = cores[i];
|
2024-03-03 11:39:46 +01:00
|
|
|
core->setKeys(core, keys);
|
2024-03-03 10:25:14 +01:00
|
|
|
core->runFrame(core);
|
|
|
|
}
|
2024-03-03 11:39:46 +01:00
|
|
|
/*hack*/
|
|
|
|
if (keys && since_last_present < FRAMESKIP_LIMIT - 8) {
|
|
|
|
since_last_present = FRAMESKIP_LIMIT - 8;
|
|
|
|
}
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-03-03 09:52:02 +01:00
|
|
|
if (!texture_updated) {
|
|
|
|
SDL_SetRenderDrawColor(renderer, 0, 0, 0, 255);
|
|
|
|
SDL_RenderClear(renderer);
|
|
|
|
|
|
|
|
/* Update SDL_Texture with last video frame (only once per new frame) */
|
|
|
|
for (i = 0; i < NUM_CHANNELS; i++) {
|
|
|
|
SDL_Surface* screen = screens[i];
|
|
|
|
SDL_Texture* texture = textures[i];
|
|
|
|
SDL_UpdateTexture(texture, NULL, screen->pixels, screen->pitch);
|
2024-12-16 03:36:56 +01:00
|
|
|
texture_updated = true;
|
2024-03-03 09:52:02 +01:00
|
|
|
SDL_RenderTexture(renderer, texture, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_RenderPresent(renderer);
|
2024-03-03 10:25:14 +01:00
|
|
|
since_last_present = 0;
|
|
|
|
} else {
|
|
|
|
since_last_present++;
|
|
|
|
if (since_last_present >= FRAMESKIP_LIMIT) {
|
|
|
|
SDL_SetRenderDrawColor(renderer, 0, 0, 0, 255);
|
|
|
|
SDL_RenderClear(renderer);
|
|
|
|
for (i = 0; i < NUM_CHANNELS; i++) {
|
|
|
|
SDL_Texture *texture = textures[i];
|
|
|
|
SDL_RenderTexture(renderer, texture, NULL, NULL);
|
|
|
|
}
|
|
|
|
SDL_RenderPresent(renderer);
|
|
|
|
}
|
2024-03-03 09:52:02 +01:00
|
|
|
}
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-12-16 03:36:56 +01:00
|
|
|
return SDL_APP_CONTINUE; /* keep iterating. */
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|
|
|
|
|
2024-12-16 03:36:56 +01:00
|
|
|
void SDL_AppQuit(void* appstate, SDL_AppResult result) {
|
2024-12-16 09:55:03 +01:00
|
|
|
struct AppState* st = (struct AppState*)appstate;
|
2024-03-03 09:52:02 +01:00
|
|
|
int i;
|
|
|
|
for (i = 0; i < NUM_CHANNELS; i++) {
|
|
|
|
mCoreConfigDeinit(&cores[i]->config);
|
|
|
|
cores[i]->deinit(cores[i]);
|
|
|
|
SDL_DestroyTexture(textures[i]);
|
|
|
|
}
|
2024-03-03 07:43:07 +01:00
|
|
|
|
2024-12-16 09:55:03 +01:00
|
|
|
SDL_CloseCamera(st->camera);
|
|
|
|
SDL_DestroyRenderer(st->renderer);
|
|
|
|
SDL_DestroyWindow(st->window);
|
|
|
|
free(st);
|
2024-03-03 07:43:07 +01:00
|
|
|
}
|