forked from mirrors/principia
184 lines
5.5 KiB
C
184 lines
5.5 KiB
C
/*
|
|
SDL_image: An example image loading library for use with SDL
|
|
Copyright (C) 1997-2024 Sam Lantinga <slouken@libsdl.org>
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
warranty. In no event will the authors be held liable for any damages
|
|
arising from the use of this software.
|
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
including commercial applications, and to alter it and redistribute it
|
|
freely, subject to the following restrictions:
|
|
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
claim that you wrote the original software. If you use this software
|
|
in a product, an acknowledgment in the product documentation would be
|
|
appreciated but is not required.
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
misrepresented as being the original software.
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
*/
|
|
|
|
/* A simple library to load images of various formats as SDL surfaces */
|
|
|
|
#include "SDL_image.h"
|
|
|
|
#if defined(SDL_BUILD_MAJOR_VERSION) && defined(SDL_COMPILE_TIME_ASSERT)
|
|
SDL_COMPILE_TIME_ASSERT(SDL_BUILD_MAJOR_VERSION,
|
|
SDL_IMAGE_MAJOR_VERSION == SDL_BUILD_MAJOR_VERSION);
|
|
SDL_COMPILE_TIME_ASSERT(SDL_BUILD_MINOR_VERSION,
|
|
SDL_IMAGE_MINOR_VERSION == SDL_BUILD_MINOR_VERSION);
|
|
SDL_COMPILE_TIME_ASSERT(SDL_BUILD_MICRO_VERSION,
|
|
SDL_IMAGE_PATCHLEVEL == SDL_BUILD_MICRO_VERSION);
|
|
#endif
|
|
|
|
#if defined(SDL_COMPILE_TIME_ASSERT)
|
|
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_MAJOR_VERSION_min, SDL_IMAGE_MAJOR_VERSION >= 0);
|
|
/* Limited only by the need to fit in SDL_version */
|
|
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_MAJOR_VERSION_max, SDL_IMAGE_MAJOR_VERSION <= 255);
|
|
|
|
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_MINOR_VERSION_min, SDL_IMAGE_MINOR_VERSION >= 0);
|
|
/* Limited only by the need to fit in SDL_version */
|
|
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_MINOR_VERSION_max, SDL_IMAGE_MINOR_VERSION <= 255);
|
|
|
|
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_PATCHLEVEL_min, SDL_IMAGE_PATCHLEVEL >= 0);
|
|
/* Limited by its encoding in SDL_VERSIONNUM and in the ABI versions */
|
|
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_PATCHLEVEL_max, SDL_IMAGE_PATCHLEVEL <= 99);
|
|
#endif
|
|
|
|
/* Table of image detection and loading functions */
|
|
static struct {
|
|
const char *type;
|
|
int (SDLCALL *is)(SDL_RWops *src);
|
|
SDL_Surface *(SDLCALL *load)(SDL_RWops *src);
|
|
} supported[] = {
|
|
{ "JPG", IMG_isJPG, IMG_LoadJPG_RW },
|
|
{ "PNG", IMG_isPNG, IMG_LoadPNG_RW },
|
|
};
|
|
|
|
const SDL_version *IMG_Linked_Version(void)
|
|
{
|
|
static SDL_version linked_version;
|
|
SDL_IMAGE_VERSION(&linked_version);
|
|
return(&linked_version);
|
|
}
|
|
|
|
extern int IMG_InitJPG(void);
|
|
extern void IMG_QuitJPG(void);
|
|
extern int IMG_InitPNG(void);
|
|
extern void IMG_QuitPNG(void);
|
|
|
|
static int initialized = 0;
|
|
|
|
int IMG_Init(int flags)
|
|
{
|
|
int result = 0;
|
|
|
|
if (flags & IMG_INIT_JPG) {
|
|
if ((initialized & IMG_INIT_JPG) || IMG_InitJPG() == 0) {
|
|
result |= IMG_INIT_JPG;
|
|
}
|
|
}
|
|
if (flags & IMG_INIT_PNG) {
|
|
if ((initialized & IMG_INIT_PNG) || IMG_InitPNG() == 0) {
|
|
result |= IMG_INIT_PNG;
|
|
}
|
|
}
|
|
|
|
initialized |= result;
|
|
|
|
return initialized;
|
|
}
|
|
|
|
void IMG_Quit()
|
|
{
|
|
if (initialized & IMG_INIT_JPG) {
|
|
IMG_QuitJPG();
|
|
}
|
|
if (initialized & IMG_INIT_PNG) {
|
|
IMG_QuitPNG();
|
|
}
|
|
|
|
initialized = 0;
|
|
}
|
|
|
|
/* Load an image from a file */
|
|
SDL_Surface *IMG_Load(const char *file)
|
|
{
|
|
SDL_RWops *src = SDL_RWFromFile(file, "rb");
|
|
const char *ext = SDL_strrchr(file, '.');
|
|
if (ext) {
|
|
ext++;
|
|
}
|
|
if (!src) {
|
|
/* The error message has been set in SDL_RWFromFile */
|
|
return NULL;
|
|
}
|
|
return IMG_LoadTyped_RW(src, 1, ext);
|
|
}
|
|
|
|
/* Load an image from an SDL datasource (for compatibility) */
|
|
SDL_Surface *IMG_Load_RW(SDL_RWops *src, int freesrc)
|
|
{
|
|
return IMG_LoadTyped_RW(src, freesrc, NULL);
|
|
}
|
|
|
|
/* Portable case-insensitive string compare function */
|
|
static int IMG_string_equals(const char *str1, const char *str2)
|
|
{
|
|
while ( *str1 && *str2 ) {
|
|
if ( SDL_toupper((unsigned char)*str1) !=
|
|
SDL_toupper((unsigned char)*str2) )
|
|
break;
|
|
++str1;
|
|
++str2;
|
|
}
|
|
return (!*str1 && !*str2);
|
|
}
|
|
|
|
/* Load an image from an SDL datasource, optionally specifying the type */
|
|
SDL_Surface *IMG_LoadTyped_RW(SDL_RWops *src, int freesrc, const char *type)
|
|
{
|
|
int i;
|
|
SDL_Surface *image;
|
|
|
|
/* Make sure there is something to do.. */
|
|
if ( src == NULL ) {
|
|
IMG_SetError("Passed a NULL data source");
|
|
return(NULL);
|
|
}
|
|
|
|
/* See whether or not this data source can handle seeking */
|
|
if ( SDL_RWseek(src, 0, RW_SEEK_CUR) < 0 ) {
|
|
IMG_SetError("Can't seek in this data source");
|
|
if (freesrc)
|
|
SDL_RWclose(src);
|
|
return(NULL);
|
|
}
|
|
|
|
/* Detect the type of image being loaded */
|
|
for ( i=0; i < SDL_arraysize(supported); ++i ) {
|
|
if (supported[i].is) {
|
|
if (!supported[i].is(src))
|
|
continue;
|
|
} else {
|
|
/* magicless format */
|
|
if (!type || !IMG_string_equals(type, supported[i].type))
|
|
continue;
|
|
}
|
|
#ifdef DEBUG_IMGLIB
|
|
fprintf(stderr, "IMGLIB: Loading image as %s\n",
|
|
supported[i].type);
|
|
#endif
|
|
image = supported[i].load(src);
|
|
if (freesrc)
|
|
SDL_RWclose(src);
|
|
return image;
|
|
}
|
|
|
|
if ( freesrc ) {
|
|
SDL_RWclose(src);
|
|
}
|
|
IMG_SetError("Unsupported image format");
|
|
return NULL;
|
|
}
|