485 lines
11 KiB
C
Raw Normal View History

2015-06-28 15:51:08 +02:00
#include <stdio.h>
2015-09-14 03:53:15 +02:00
#include <stdint.h>
2015-06-28 15:51:08 +02:00
#include <stdlib.h>
#include <string.h>
2015-09-14 03:53:15 +02:00
#include <assert.h>
2015-06-28 15:51:08 +02:00
#include <boolean.h>
#include <lists/dir_list.h>
2016-01-20 15:40:08 -05:00
#include <file/file_path.h>
#include <compat/strl.h>
2016-09-04 21:34:03 +02:00
#include <retro_environment.h>
2015-06-28 15:51:08 +02:00
2016-09-04 21:39:16 +02:00
#if defined(HAVE_RPNG) || defined(HAVE_RJPEG) || defined(HAVE_RTGA) || defined(HAVE_RBMP)
#define PREFER_NON_STB_IMAGE
#endif
#if defined(HAVE_STB_IMAGE) && !defined(PREFER_NON_STB_IMAGE)
2015-06-28 15:51:08 +02:00
#define STB_IMAGE_IMPLEMENTATION
#if 0
2015-06-28 15:51:08 +02:00
#define STBI_NO_PSD
#define STBI_NO_GIF
#define STBI_NO_HDR
#define STBI_NO_PIC
#define STBI_NO_PNM
#endif
#define STBI_SUPPORT_ZLIB
2015-06-28 15:51:08 +02:00
#ifdef RARCH_INTERNAL
2016-01-21 19:53:52 +01:00
#include "../../deps/stb/stb_image.h"
#else
#include <stb_image.h>
#endif
#else
#include <formats/image.h>
#endif
2015-06-28 15:51:08 +02:00
#include <libretro.h>
2015-06-28 15:51:08 +02:00
#ifdef RARCH_INTERNAL
2015-09-06 14:59:20 +02:00
#include "internal_cores.h"
2015-06-28 15:51:08 +02:00
#define IMAGE_CORE_PREFIX(s) libretro_imageviewer_##s
#else
#define IMAGE_CORE_PREFIX(s) s
#endif
static retro_log_printf_t IMAGE_CORE_PREFIX(log_cb);
static retro_video_refresh_t IMAGE_CORE_PREFIX(video_cb);
static retro_input_poll_t IMAGE_CORE_PREFIX(input_poll_cb);
static retro_input_state_t IMAGE_CORE_PREFIX(input_state_cb);
static retro_audio_sample_batch_t IMAGE_CORE_PREFIX(audio_batch_cb);
static retro_environment_t IMAGE_CORE_PREFIX(environ_cb);
2016-02-22 16:55:02 +01:00
static bool process_new_image;
2015-06-28 15:51:08 +02:00
static uint32_t* image_buffer;
2016-09-04 21:34:03 +02:00
#ifndef STB_IMAGE_IMPLEMENTATION
static struct texture_image image_texture;
#endif
2015-06-28 15:51:08 +02:00
static int image_width;
static int image_height;
static bool image_uploaded;
static bool slideshow_enable;
2016-01-20 15:40:08 -05:00
struct string_list *file_list;
2015-06-28 15:51:08 +02:00
#if 0
#define DUPE_TEST
#endif
2016-09-04 21:34:03 +02:00
#ifdef STB_IMAGE_IMPLEMENTATION
2016-01-20 15:40:08 -05:00
static const char* IMAGE_CORE_PREFIX(valid_extensions) = "jpg|jpeg|png|bmp|psd|tga|gif|hdr|pic|ppm|pgm";
#else
2016-10-01 19:46:49 +02:00
static const char* IMAGE_CORE_PREFIX(valid_extensions) = 1+ /* to remove the first |, the alternative is 25 extra lines of ifdef/etc */
2016-10-01 19:32:02 +02:00
#ifdef HAVE_RJPEG
2016-10-01 19:46:49 +02:00
"|jpg|jpeg"
#endif
#ifdef HAVE_RPNG
2016-10-01 19:46:49 +02:00
"|png"
#endif
#ifdef HAVE_RBMP
2016-10-01 19:46:49 +02:00
"|bmp"
#endif
#ifdef HAVE_RTGA
2016-10-01 19:46:49 +02:00
"|tga"
2016-10-01 19:32:02 +02:00
#endif
2016-10-01 19:46:49 +02:00
#if !defined(HAVE_RJPEG) && !defined(HAVE_RPNG) && !defined(HAVE_RBMP) && !defined(HAVE_RTGA)
2016-10-01 19:32:02 +02:00
#error "can't build this core with no image formats"
#endif
2016-10-01 19:32:02 +02:00
;
2016-01-20 15:40:08 -05:00
#endif
2015-06-28 15:51:08 +02:00
void IMAGE_CORE_PREFIX(retro_get_system_info)(struct retro_system_info *info)
{
info->library_name = "image display";
info->library_version = "v0.1";
info->need_fullpath = true;
info->block_extract = false;
2016-01-20 15:40:08 -05:00
info->valid_extensions = IMAGE_CORE_PREFIX(valid_extensions);
2015-06-28 15:51:08 +02:00
}
void IMAGE_CORE_PREFIX(retro_get_system_av_info)(struct retro_system_av_info *info)
{
info->geometry.base_width = image_width;
info->geometry.base_height = image_height;
info->geometry.max_width = image_width;
info->geometry.max_height = image_height;
info->geometry.aspect_ratio = 0;
info->timing.fps = 60.0;
info->timing.sample_rate = 44100.0;
}
static void imageviewer_reset(void)
{
image_buffer = NULL;
image_width = 0;
image_height = 0;
}
2015-06-28 15:51:08 +02:00
void IMAGE_CORE_PREFIX(retro_init)(void)
{
struct retro_log_callback log;
if (IMAGE_CORE_PREFIX(environ_cb)(RETRO_ENVIRONMENT_GET_LOG_INTERFACE, &log))
IMAGE_CORE_PREFIX(log_cb) = log.log;
else
IMAGE_CORE_PREFIX(log_cb) = NULL;
imageviewer_reset();
2015-06-28 15:51:08 +02:00
}
static void imageviewer_free_image(void)
2015-06-28 15:51:08 +02:00
{
2016-09-04 21:34:03 +02:00
#ifdef STB_IMAGE_IMPLEMENTATION
2015-06-28 15:51:08 +02:00
if (image_buffer)
free(image_buffer);
#else
image_texture_free(&image_texture);
#endif
2015-06-28 15:51:08 +02:00
image_buffer = NULL;
}
void IMAGE_CORE_PREFIX(retro_deinit)(void)
{
imageviewer_free_image();
imageviewer_reset();
2015-06-28 15:51:08 +02:00
}
void IMAGE_CORE_PREFIX(retro_set_environment)(retro_environment_t cb)
{
static const struct retro_variable vars[] = {
{ NULL, NULL },
};
IMAGE_CORE_PREFIX(environ_cb) = cb;
cb(RETRO_ENVIRONMENT_SET_VARIABLES, (void*)vars);
}
void IMAGE_CORE_PREFIX(retro_set_video_refresh)(retro_video_refresh_t cb)
{
IMAGE_CORE_PREFIX(video_cb) = cb;
}
void IMAGE_CORE_PREFIX(retro_set_audio_sample)(retro_audio_sample_t unused)
{
}
void IMAGE_CORE_PREFIX(retro_set_audio_sample_batch)(retro_audio_sample_batch_t cb)
{
IMAGE_CORE_PREFIX(audio_batch_cb) = cb;
}
void IMAGE_CORE_PREFIX(retro_set_input_poll)(retro_input_poll_t cb)
{
IMAGE_CORE_PREFIX(input_poll_cb) = cb;
}
void IMAGE_CORE_PREFIX(retro_set_input_state)(retro_input_state_t cb)
{
IMAGE_CORE_PREFIX(input_state_cb) = cb;
}
void IMAGE_CORE_PREFIX(retro_set_controller_port_device)(unsigned a, unsigned b)
{
}
void IMAGE_CORE_PREFIX(retro_reset)(void)
{
image_uploaded = false;
}
size_t IMAGE_CORE_PREFIX(retro_serialize_size)(void)
{
return 0;
}
bool IMAGE_CORE_PREFIX(retro_serialize)(void *data, size_t size)
{
(void)data;
(void)size;
return false;
}
bool IMAGE_CORE_PREFIX(retro_unserialize)(const void *data, size_t size)
{
(void)data;
(void)size;
return false;
}
void IMAGE_CORE_PREFIX(retro_cheat_reset)(void)
{
}
void IMAGE_CORE_PREFIX(retro_cheat_set)(unsigned a, bool b, const char * c)
{
}
2016-02-22 16:55:02 +01:00
static bool imageviewer_load(const char *path, int image_index)
{
2016-09-04 21:34:03 +02:00
#ifdef STB_IMAGE_IMPLEMENTATION
int comp;
2016-07-31 14:29:33 +02:00
#endif
2017-01-09 01:55:38 +01:00
#ifdef RARCH_INTERNAL
extern bool video_driver_supports_rgba(void);
#endif
imageviewer_free_image();
2016-09-04 21:34:03 +02:00
#ifdef STB_IMAGE_IMPLEMENTATION
image_buffer = (uint32_t*)stbi_load(
path,
&image_width,
&image_height,
&comp,
4);
#else
#ifdef RARCH_INTERNAL
image_texture.supports_rgba = video_driver_supports_rgba();
#endif
if (!image_texture_load(&image_texture, path))
return false;
image_buffer = (uint32_t*)image_texture.pixels;
image_width = image_texture.width;
image_height = image_texture.height;
#endif
2016-02-22 16:55:02 +01:00
if (!image_buffer)
return false;
2016-02-22 16:55:02 +01:00
process_new_image = true;
return true;
}
2015-06-28 15:51:08 +02:00
bool IMAGE_CORE_PREFIX(retro_load_game)(const struct retro_game_info *info)
{
enum retro_pixel_format fmt = RETRO_PIXEL_FORMAT_XRGB8888;
char *dir = strdup(info->path);
slideshow_enable = false;
2016-01-20 15:40:08 -05:00
path_basedir(dir);
file_list = dir_list_new(dir, IMAGE_CORE_PREFIX(valid_extensions),
false,true,false,false);
2016-01-20 15:40:08 -05:00
dir_list_sort(file_list, false);
free(dir);
2015-06-28 15:51:08 +02:00
if (!IMAGE_CORE_PREFIX(environ_cb)(RETRO_ENVIRONMENT_SET_PIXEL_FORMAT, &fmt))
{
if (IMAGE_CORE_PREFIX(log_cb))
IMAGE_CORE_PREFIX(log_cb)(RETRO_LOG_INFO, "XRGB8888 is not supported.\n");
return false;
}
2016-02-22 16:55:02 +01:00
if (!imageviewer_load(info->path, 0))
return false;
2015-06-28 15:51:08 +02:00
return true;
}
bool IMAGE_CORE_PREFIX(retro_load_game_special)(unsigned a, const struct retro_game_info *b, size_t c)
{
return false;
}
2015-06-28 15:51:08 +02:00
void IMAGE_CORE_PREFIX(retro_unload_game)(void)
{
imageviewer_free_image();
2015-06-28 15:51:08 +02:00
image_width = 0;
image_height = 0;
}
unsigned IMAGE_CORE_PREFIX(retro_get_region)(void)
{
return RETRO_REGION_NTSC;
}
void *IMAGE_CORE_PREFIX(retro_get_memory_data)(unsigned id)
{
return NULL;
}
size_t IMAGE_CORE_PREFIX(retro_get_memory_size)(unsigned id)
{
return 0;
}
2016-01-20 15:40:08 -05:00
void IMAGE_CORE_PREFIX(retro_run)(void)
{
bool first_image = false;
bool last_image = false;
bool backwards_image = false;
bool forward_image = false;
2016-01-21 20:13:24 +01:00
bool load_image = false;
bool next_image = false;
bool prev_image = false;
static int frames = 0;
2016-01-20 15:40:08 -05:00
static int image_index = 0;
2016-01-23 15:28:11 +01:00
uint16_t input = 0;
static uint16_t previnput;
uint16_t realinput = 0;
2016-01-23 15:28:11 +01:00
int i;
2016-01-20 15:40:08 -05:00
2015-06-28 15:51:08 +02:00
IMAGE_CORE_PREFIX(input_poll_cb)();
if (slideshow_enable)
2016-01-20 15:40:08 -05:00
{
2016-02-28 22:51:02 +01:00
if ((frames % 120 == 0) && image_index < (signed)(file_list->size - 1))
next_image = true;
}
2016-01-20 15:40:08 -05:00
2016-01-23 15:28:11 +01:00
for (i=0;i<16;i++)
{
if (IMAGE_CORE_PREFIX(input_state_cb)(0, RETRO_DEVICE_JOYPAD, 0, i))
{
realinput |= 1<<i;
2016-01-23 15:28:11 +01:00
}
}
input = realinput & ~previnput;
previnput = realinput;
2016-01-23 15:28:11 +01:00
if (input & (1<<RETRO_DEVICE_ID_JOYPAD_UP))
{
2016-02-28 22:51:02 +01:00
if ((image_index + 5) < (signed)(file_list->size - 1))
forward_image = true;
else
last_image = true;
}
2016-01-23 15:28:11 +01:00
if (input & (1<<RETRO_DEVICE_ID_JOYPAD_DOWN))
{
if ((image_index - 5) > 0)
backwards_image = true;
else
first_image = true;
}
2016-01-23 15:28:11 +01:00
if (input & (1<<RETRO_DEVICE_ID_JOYPAD_LEFT))
{
if (image_index > 0)
prev_image = true;
}
2016-01-23 15:28:11 +01:00
if (input & (1<<RETRO_DEVICE_ID_JOYPAD_RIGHT))
{
2016-02-28 22:51:02 +01:00
if (image_index < (signed)(file_list->size - 1))
next_image = true;
}
2015-06-28 15:51:08 +02:00
2016-01-23 15:28:11 +01:00
if (input & (1<<RETRO_DEVICE_ID_JOYPAD_Y))
{
slideshow_enable = !slideshow_enable;
}
2016-01-20 15:40:08 -05:00
if (prev_image)
{
2016-01-21 20:13:24 +01:00
image_index--;
load_image = true;
2016-01-20 15:40:08 -05:00
}
else if (next_image)
{
2016-01-21 20:13:24 +01:00
image_index++;
load_image = true;
}
else if (backwards_image)
{
image_index -= 5;
load_image = true;
}
else if (forward_image)
{
image_index += 5;
load_image = true;
}
else if (first_image)
{
image_index = 0;
load_image = true;
}
else if (last_image)
{
image_index = file_list->size - 1;
load_image = true;
}
2016-01-21 20:13:24 +01:00
if (load_image)
{
2016-02-22 16:55:02 +01:00
if (!imageviewer_load(file_list->elems[image_index].data, image_index))
{
IMAGE_CORE_PREFIX(environ_cb)(RETRO_ENVIRONMENT_SHUTDOWN, NULL);
}
}
2016-01-21 20:13:24 +01:00
2016-02-22 16:55:02 +01:00
if (process_new_image)
{
/* RGBA > XRGB8888 */
struct retro_system_av_info info;
2016-09-04 21:34:03 +02:00
#ifdef STB_IMAGE_IMPLEMENTATION
int x, y;
uint32_t *buf = &image_buffer[0];
for (y = 0; y < image_height; y++)
2016-02-22 16:55:02 +01:00
{
for (x = 0; x < image_width; x++, buf++)
{
uint32_t pixel = *buf;
uint32_t a = pixel >> 24;
if (a == 255)
*buf = (pixel & 0x0000ff00) | ((pixel << 16) & 0x00ff0000) | ((pixel >> 16) & 0x000000ff);
else
{
uint32_t r = pixel & 0x0000ff;
uint32_t g = (pixel & 0x00ff00) >> 8;
uint32_t b = (pixel & 0xff0000) >> 16;
uint32_t bg = ((x & 8) ^ (y & 8)) ? 0x66 : 0x99;
r = a * r / 255 + (255 - a) * bg / 255;
g = a * g / 255 + (255 - a) * bg / 255;
b = a * b / 255 + (255 - a) * bg / 255;
*buf = r << 16 | g << 8 | b;
}
}
2016-02-22 16:55:02 +01:00
}
#endif
2016-02-22 16:55:02 +01:00
IMAGE_CORE_PREFIX(retro_get_system_av_info)(&info);
IMAGE_CORE_PREFIX(environ_cb)(RETRO_ENVIRONMENT_SET_GEOMETRY, &info.geometry);
process_new_image = false;
}
2015-06-28 15:51:08 +02:00
#ifdef DUPE_TEST
if (!image_uploaded)
{
IMAGE_CORE_PREFIX(video_cb)(image_buffer, image_width, image_height, image_width * sizeof(uint32_t));
image_uploaded = true;
}
else
IMAGE_CORE_PREFIX(video_cb)(NULL, image_width, image_height, image_width * sizeof(uint32_t));
#else
IMAGE_CORE_PREFIX(video_cb)(image_buffer, image_width, image_height, image_width * sizeof(uint32_t));
#endif
2016-01-20 15:40:08 -05:00
frames++;
2015-06-28 15:51:08 +02:00
}
unsigned IMAGE_CORE_PREFIX(retro_api_version)(void)
{
return RETRO_API_VERSION;
}