123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227 |
- /*
- Simple DirectMedia Layer
- Copyright (C) 1997-2025 Sam Lantinga <[email protected]>
- 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.
- */
- /* WIKI CATEGORY: DlopenNotes */
- /**
- * # CategoryDlopenNotes
- *
- * This header allows you to annotate your code so external tools know about
- * dynamic shared library dependencies.
- *
- * If you determine that your toolchain doesn't support dlopen notes, you can
- * disable this feature by defining `SDL_DISABLE_DLOPEN_NOTES`. You can use
- * this CMake snippet to check for support:
- *
- * ```cmake
- * set(CHECK_ELF_DLNOTES_SRC [==[
- * #ifndef __ELF__
- * ELF DL notes is only supported on ELF platforms
- * #endif
- * __attribute__ ((used,aligned(4),section(".note.dlopen"))) static const struct {
- * struct { int a; int b; int c; } hdr; char name[4]; __attribute__((aligned(4))) char json[24];
- * } dlnote = { { 4, 0x407c0c0aU, 16 }, "FDO", "[\\"a\\":{\\"a\\":\\"1\\",\\"b\\":\\"2\\"}]" };
- * int main(int argc, char *argv[]) {
- * return argc + dlnote.hdr.a;
- * }
- * ]==])
- * check_c_source_compiles("${CHECK_ELF_DLNOTES_SRC}" COMPILER_SUPPORTS_ELFNOTES)
- * if(NOT COMPILER_SUPPORTS_ELFNOTES)
- * set(SDL_DISABLE_DLOPEN_NOTES TRUE)
- * endif()
- * ```
- */
- #ifndef SDL_dlopennote_h
- #define SDL_dlopennote_h
- /**
- * Use this macro with SDL_ELF_NOTE_DLOPEN() to note that a dynamic shared
- * library dependency is optional.
- *
- * Optional functionality uses the dependency, the binary will work and the
- * dependency is only needed for full-featured installations.
- *
- * \since This macro is available since SDL 3.4.0.
- *
- * \sa SDL_ELF_NOTE_DLOPEN
- * \sa SDL_ELF_NOTE_DLOPEN_PRIORITY_RECOMMENDED
- * \sa SDL_ELF_NOTE_DLOPEN_PRIORITY_REQUIRED
- */
- #define SDL_ELF_NOTE_DLOPEN_PRIORITY_SUGGESTED "suggested"
- /**
- * Use this macro with SDL_ELF_NOTE_DLOPEN() to note that a dynamic shared
- * library dependency is recommended.
- *
- * Important functionality needs the dependency, the binary will work but in
- * most cases the dependency should be provided.
- *
- * \since This macro is available since SDL 3.4.0.
- *
- * \sa SDL_ELF_NOTE_DLOPEN
- * \sa SDL_ELF_NOTE_DLOPEN_PRIORITY_SUGGESTED
- * \sa SDL_ELF_NOTE_DLOPEN_PRIORITY_REQUIRED
- */
- #define SDL_ELF_NOTE_DLOPEN_PRIORITY_RECOMMENDED "recommended"
- /**
- * Use this macro with SDL_ELF_NOTE_DLOPEN() to note that a dynamic shared
- * library dependency is required.
- *
- * Core functionality needs the dependency, the binary will not work if it
- * cannot be found.
- *
- * \since This macro is available since SDL 3.4.0.
- *
- * \sa SDL_ELF_NOTE_DLOPEN
- * \sa SDL_ELF_NOTE_DLOPEN_PRIORITY_SUGGESTED
- * \sa SDL_ELF_NOTE_DLOPEN_PRIORITY_RECOMMENDED
- */
- #define SDL_ELF_NOTE_DLOPEN_PRIORITY_REQUIRED "required"
- #if !defined(SDL_PLATFORM_UNIX) || defined(SDL_PLATFORM_ANDROID)
- /* The dlopen note functionality isn't used on this platform */
- #ifndef SDL_DISABLE_DLOPEN_NOTES
- #define SDL_DISABLE_DLOPEN_NOTES
- #endif
- #elif defined(__GNUC__) && (__GNUC__ < 3 || (__GNUC__ == 3 && __GNUC_MINOR__ < 1))
- /* gcc < 3.1 too old */
- #ifndef SDL_DISABLE_DLOPEN_NOTES
- #define SDL_DISABLE_DLOPEN_NOTES
- #endif
- #endif /* SDL_PLATFORM_UNIX || SDL_PLATFORM_ANDROID */
- #if defined(__ELF__) && !defined(SDL_DISABLE_DLOPEN_NOTES)
- #include <SDL3/SDL_stdinc.h>
- #define SDL_ELF_NOTE_DLOPEN_VENDOR "FDO"
- #define SDL_ELF_NOTE_DLOPEN_TYPE 0x407c0c0aU
- #define SDL_ELF_NOTE_INTERNAL2(json, variable_name) \
- __attribute__((aligned(4), used, section(".note.dlopen"))) \
- static const struct { \
- struct { \
- Uint32 n_namesz; \
- Uint32 n_descsz; \
- Uint32 n_type; \
- } nhdr; \
- char name[4]; \
- __attribute__((aligned(4))) char dlopen_json[sizeof(json)]; \
- } variable_name = { \
- { \
- sizeof(SDL_ELF_NOTE_DLOPEN_VENDOR), \
- sizeof(json), \
- SDL_ELF_NOTE_DLOPEN_TYPE \
- }, \
- SDL_ELF_NOTE_DLOPEN_VENDOR, \
- json \
- }
- #define SDL_ELF_NOTE_INTERNAL(json, variable_name) \
- SDL_ELF_NOTE_INTERNAL2(json, variable_name)
- #define SDL_SONAME_ARRAY1(N1) "[\"" N1 "\"]"
- #define SDL_SONAME_ARRAY2(N1,N2) "[\"" N1 "\",\"" N2 "\"]"
- #define SDL_SONAME_ARRAY3(N1,N2,N3) "[\"" N1 "\",\"" N2 "\",\"" N3 "\"]"
- #define SDL_SONAME_ARRAY4(N1,N2,N3,N4) "[\"" N1 "\",\"" N2 "\",\"" N3 "\",\"" N4 "\"]"
- #define SDL_SONAME_ARRAY5(N1,N2,N3,N4,N5) "[\"" N1 "\",\"" N2 "\",\"" N3 "\",\"" N4 "\",\"" N5 "\"]"
- #define SDL_SONAME_ARRAY6(N1,N2,N3,N4,N5,N6) "[\"" N1 "\",\"" N2 "\",\"" N3 "\",\"" N4 "\",\"" N5 "\",\"" N6 "\"]"
- #define SDL_SONAME_ARRAY7(N1,N2,N3,N4,N5,N6,N7) "[\"" N1 "\",\"" N2 "\",\"" N3 "\",\"" N4 "\",\"" N5 "\",\"" N6 "\",\"" N7 "\"]"
- #define SDL_SONAME_ARRAY8(N1,N2,N3,N4,N5,N6,N7,N8) "[\"" N1 "\",\"" N2 "\",\"" N3 "\",\"" N4 "\",\"" N5 "\",\"" N6 "\",\"" N7 "\",\"" N8 "\"]"
- #define SDL_SONAME_ARRAY_GET(N1,N2,N3,N4,N5,N6,N7,N8,NAME,...) NAME
- #define SDL_SONAME_ARRAY(...) \
- SDL_SONAME_ARRAY_GET(__VA_ARGS__, \
- SDL_SONAME_ARRAY8, \
- SDL_SONAME_ARRAY7, \
- SDL_SONAME_ARRAY6, \
- SDL_SONAME_ARRAY5, \
- SDL_SONAME_ARRAY4, \
- SDL_SONAME_ARRAY3, \
- SDL_SONAME_ARRAY2, \
- SDL_SONAME_ARRAY1 \
- )(__VA_ARGS__)
- /* Create "unique" variable name using __LINE__,
- * so creating elf notes on the same line is not supported
- */
- #define SDL_ELF_NOTE_JOIN2(A,B) A##B
- #define SDL_ELF_NOTE_JOIN(A,B) SDL_ELF_NOTE_JOIN2(A,B)
- #define SDL_ELF_NOTE_UNIQUE_NAME SDL_ELF_NOTE_JOIN(s_SDL_dlopen_note_, __LINE__)
- /**
- * Note that your application has dynamic shared library dependencies.
- *
- * You can do this by adding the following to the global scope:
- *
- * ```c
- * SDL_ELF_NOTE_DLOPEN(
- * "png",
- * "Support for loading PNG images using libpng (required for APNG)",
- * SDL_ELF_NOTE_DLOPEN_PRIORITY_RECOMMENDED,
- * "libpng12.so.0"
- * );
- * ```
- *
- * Or if you support multiple versions of a library, you can list them:
- *
- * ```c
- * // Our app supports SDL1, SDL2, and SDL3 by dynamically loading them
- * SDL_ELF_NOTE_DLOPEN(
- * "SDL",
- * "Create windows through SDL video backend",
- * SDL_ELF_NOTE_DLOPEN_PRIORITY_REQUIRED
- * "libSDL-1.2.so.0", "libSDL2-2.0.so.0", "libSDL3.so.0"
- * );
- * ```
- *
- * \since This macro is available since SDL 3.4.0.
- *
- * \sa SDL_ELF_NOTE_DLOPEN_PRIORITY_SUGGESTED
- * \sa SDL_ELF_NOTE_DLOPEN_PRIORITY_RECOMMENDED
- * \sa SDL_ELF_NOTE_DLOPEN_PRIORITY_REQUIRED
- */
- #define SDL_ELF_NOTE_DLOPEN(feature, description, priority, ...) \
- SDL_ELF_NOTE_INTERNAL( \
- "[{\"feature\":\"" feature \
- "\",\"description\":\"" description \
- "\",\"priority\":\"" priority \
- "\",\"soname\":" SDL_SONAME_ARRAY(__VA_ARGS__) "}]", \
- SDL_ELF_NOTE_UNIQUE_NAME)
- #elif defined(__GNUC__) && __GNUC__ < 3
- #define SDL_ELF_NOTE_DLOPEN(args...)
- #elif defined(_MSC_VER) && _MSC_VER < 1400
- /* Variadic macros are not supported */
- #define SDL_ELF_NOTE_DLOPEN
- #else
- #define SDL_ELF_NOTE_DLOPEN(...)
- #endif
- #endif /* SDL_dlopennote_h */
|