123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467 |
- #!/usr/bin/env python3
- # Simple DirectMedia Layer
- # Copyright (C) 1997-2022 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.
- # WHAT IS THIS?
- # When you add a public API to SDL, please run this script, make sure the
- # output looks sane (git diff, it adds to existing files), and commit it.
- # It keeps the dynamic API jump table operating correctly.
- #
- # OS-specific API:
- # After running the script, you have to manually add #ifdef __WIN32__
- # or similar around the function in 'SDL_dynapi_procs.h'
- #
- import re
- import os
- import argparse
- import pprint
- import json
- dir_path = os.path.dirname(os.path.realpath(__file__))
- sdl_include_dir = dir_path + "/../../include/SDL3/"
- sdl_dynapi_procs_h = dir_path + "/../../src/dynapi/SDL_dynapi_procs.h"
- sdl_dynapi_overrides_h = dir_path + "/../../src/dynapi/SDL_dynapi_overrides.h"
- sdl_dynapi_sym = dir_path + "/../../src/dynapi/SDL_dynapi.sym"
- full_API = []
- def main():
- # Parse 'sdl_dynapi_procs_h' file to find existing functions
- existing_procs = find_existing_procs()
- # Get list of SDL headers
- sdl_list_includes = get_header_list()
- reg_externC = re.compile('.*extern[ "]*C[ "].*')
- reg_comment_remove_content = re.compile('\/\*.*\*/')
- reg_parsing_function = re.compile('(.*SDLCALL[^\(\)]*) ([a-zA-Z0-9_]+) *\((.*)\) *;.*')
- #eg:
- # void (SDLCALL *callback)(void*, int)
- # \1(\2)\3
- reg_parsing_callback = re.compile('([^\(\)]*)\(([^\(\)]+)\)(.*)')
- for filename in sdl_list_includes:
- if args.debug:
- print("Parse header: %s" % filename)
- input = open(filename)
- parsing_function = False
- current_func = ""
- parsing_comment = False
- current_comment = ""
- for line in input:
- # Discard pre-processor directives ^#.*
- if line.startswith("#"):
- continue
- # Discard "extern C" line
- match = reg_externC.match(line)
- if match:
- continue
- # Remove one line comment /* ... */
- # eg: extern DECLSPEC SDL_hid_device * SDLCALL SDL_hid_open_path(const char *path, int bExclusive /* = false */);
- line = reg_comment_remove_content.sub('', line)
- # Get the comment block /* ... */ across several lines
- match_start = "/*" in line
- match_end = "*/" in line
- if match_start and match_end:
- continue
- if match_start:
- parsing_comment = True
- current_comment = line
- continue
- if match_end:
- parsing_comment = False
- current_comment += line
- continue
- if parsing_comment:
- current_comment += line
- continue
- # Get the function prototype across several lines
- if parsing_function:
- # Append to the current function
- current_func += " "
- current_func += line.strip()
- else:
- # if is contains "extern", start grabbing
- if "extern" not in line:
- continue
- # Start grabbing the new function
- current_func = line.strip()
- parsing_function = True;
- # If it contains ';', then the function is complete
- if ";" not in current_func:
- continue
- # Got function/comment, reset vars
- parsing_function = False;
- func = current_func
- comment = current_comment
- current_func = ""
- current_comment = ""
- # Discard if it doesn't contain 'SDLCALL'
- if "SDLCALL" not in func:
- if args.debug:
- print(" Discard: " + func)
- continue
- if args.debug:
- print(" Raw data: " + func);
- # Replace unusual stuff...
- func = func.replace("SDL_PRINTF_VARARG_FUNC(1)", "");
- func = func.replace("SDL_PRINTF_VARARG_FUNC(2)", "");
- func = func.replace("SDL_PRINTF_VARARG_FUNC(3)", "");
- func = func.replace("SDL_SCANF_VARARG_FUNC(2)", "");
- func = func.replace("__attribute__((analyzer_noreturn))", "");
- func = func.replace("SDL_MALLOC", "");
- func = func.replace("SDL_ALLOC_SIZE2(1, 2)", "");
- func = func.replace("SDL_ALLOC_SIZE(2)", "");
- # Should be a valid function here
- match = reg_parsing_function.match(func)
- if not match:
- print("Cannot parse: "+ func)
- exit(-1)
- func_ret = match.group(1)
- func_name = match.group(2)
- func_params = match.group(3)
- #
- # Parse return value
- #
- func_ret = func_ret.replace('extern', ' ')
- func_ret = func_ret.replace('SDLCALL', ' ')
- func_ret = func_ret.replace('DECLSPEC', ' ')
- # Remove trailling spaces in front of '*'
- tmp = ""
- while func_ret != tmp:
- tmp = func_ret
- func_ret = func_ret.replace(' ', ' ')
- func_ret = func_ret.replace(' *', '*')
- func_ret = func_ret.strip()
- #
- # Parse parameters
- #
- func_params = func_params.strip()
- if func_params == "":
- func_params = "void"
- # Identify each function parameters with type and name
- # (eventually there are callbacks of several parameters)
- tmp = func_params.split(',')
- tmp2 = []
- param = ""
- for t in tmp:
- if param == "":
- param = t
- else:
- param = param + "," + t
- # Identify a callback or parameter when there is same count of '(' and ')'
- if param.count('(') == param.count(')'):
- tmp2.append(param.strip())
- param = ""
- # Process each parameters, separation name and type
- func_param_type = []
- func_param_name = []
- for t in tmp2:
- if t == "void":
- func_param_type.append(t)
- func_param_name.append("")
- continue
- if t == "...":
- func_param_type.append(t)
- func_param_name.append("")
- continue
- param_name = ""
- # parameter is a callback
- if '(' in t:
- match = reg_parsing_callback.match(t)
- if not match:
- print("cannot parse callback: " + t);
- exit(-1)
- a = match.group(1).strip()
- b = match.group(2).strip()
- c = match.group(3).strip()
- try:
- (param_type, param_name) = b.rsplit('*', 1)
- except:
- param_type = t;
- param_name = "param_name_not_specified"
- # bug rsplit ??
- if param_name == "":
- param_name = "param_name_not_specified"
- # recontruct a callback name for future parsing
- func_param_type.append(a + " (" + param_type.strip() + " *REWRITE_NAME)" + c)
- func_param_name.append(param_name.strip())
- continue
- # array like "char *buf[]"
- has_array = False
- if t.endswith("[]"):
- t = t.replace("[]", "")
- has_array = True
- # pointer
- if '*' in t:
- try:
- (param_type, param_name) = t.rsplit('*', 1)
- except:
- param_type = t;
- param_name = "param_name_not_specified"
- # bug rsplit ??
- if param_name == "":
- param_name = "param_name_not_specified"
- val = param_type.strip() + "*REWRITE_NAME"
- # Remove trailling spaces in front of '*'
- tmp = ""
- while val != tmp:
- tmp = val
- val = val.replace(' ', ' ')
- val = val.replace(' *', '*')
- # first occurence
- val = val.replace('*', ' *', 1)
- val = val.strip()
- else: # non pointer
- # cut-off last word on
- try:
- (param_type, param_name) = t.rsplit(' ', 1)
- except:
- param_type = t;
- param_name = "param_name_not_specified"
- val = param_type.strip() + " REWRITE_NAME"
- # set back array
- if has_array:
- val += "[]"
- func_param_type.append(val)
- func_param_name.append(param_name.strip())
- new_proc = {}
- # Return value type
- new_proc['retval'] = func_ret
- # List of parameters (type + anonymized param name 'REWRITE_NAME')
- new_proc['parameter'] = func_param_type
- # Real parameter name, or 'param_name_not_specified'
- new_proc['parameter_name'] = func_param_name
- # Function name
- new_proc['name'] = func_name
- # Header file
- new_proc['header'] = os.path.basename(filename)
- # Function comment
- new_proc['comment'] = comment
- full_API.append(new_proc)
- if args.debug:
- pprint.pprint(new_proc);
- print("\n")
- if func_name not in existing_procs:
- print("NEW " + func)
- add_dyn_api(new_proc)
- # For-End line in input
- input.close()
- # For-End parsing all files of sdl_list_includes
- # Dump API into a json file
- full_API_json();
- # Dump API into a json file
- def full_API_json():
- if args.dump:
- filename = 'sdl.json'
- with open(filename, 'w') as f:
- json.dump(full_API, f, indent=4, sort_keys=True)
- print("dump API to '%s'" % filename);
- # Parse 'sdl_dynapi_procs_h' file to find existing functions
- def find_existing_procs():
- reg = re.compile('SDL_DYNAPI_PROC\([^,]*,([^,]*),.*\)')
- ret = []
- input = open(sdl_dynapi_procs_h)
- for line in input:
- match = reg.match(line)
- if not match:
- continue
- existing_func = match.group(1)
- ret.append(existing_func);
- # print(existing_func)
- input.close()
- return ret
- # Get list of SDL headers
- def get_header_list():
- reg = re.compile('^.*\.h$')
- ret = []
- tmp = os.listdir(sdl_include_dir)
- for f in tmp:
- # Only *.h files
- match = reg.match(f)
- if not match:
- if args.debug:
- print("Skip %s" % f)
- continue
- ret.append(sdl_include_dir + f)
- return ret
- # Write the new API in files: _procs.h _overrivides.h and .sym
- def add_dyn_api(proc):
- func_name = proc['name']
- func_ret = proc['retval']
- func_argtype = proc['parameter']
- # File: SDL_dynapi_procs.h
- #
- # Add at last
- # SDL_DYNAPI_PROC(SDL_EGLConfig,SDL_EGL_GetCurrentEGLConfig,(void),(),return)
- f = open(sdl_dynapi_procs_h, "a")
- dyn_proc = "SDL_DYNAPI_PROC(" + func_ret + "," + func_name + ",("
- i = ord('a')
- remove_last = False
- for argtype in func_argtype:
- # Special case, void has no parameter name
- if argtype == "void":
- dyn_proc += "void"
- continue
- # Var name: a, b, c, ...
- varname = chr(i)
- i += 1
- tmp = argtype.replace("REWRITE_NAME", varname)
- dyn_proc += tmp + ", "
- remove_last = True
- # remove last 2 char ', '
- if remove_last:
- dyn_proc = dyn_proc[:-1]
- dyn_proc = dyn_proc[:-1]
- dyn_proc += "),("
- i = ord('a')
- remove_last = False
- for argtype in func_argtype:
- # Special case, void has no parameter name
- if argtype == "void":
- continue
- # Special case, '...' has no parameter name
- if argtype == "...":
- continue
- # Var name: a, b, c, ...
- varname = chr(i)
- i += 1
- dyn_proc += varname + ","
- remove_last = True
- # remove last char ','
- if remove_last:
- dyn_proc = dyn_proc[:-1]
- dyn_proc += "),"
- if func_ret != "void":
- dyn_proc += "return"
- dyn_proc += ")"
- f.write(dyn_proc + "\n")
- f.close()
- # File: SDL_dynapi_overrides.h
- #
- # Add at last
- # "#define SDL_DelayNS SDL_DelayNS_REAL
- f = open(sdl_dynapi_overrides_h, "a")
- f.write("#define " + func_name + " " + func_name + "_REAL\n")
- f.close()
- # File: SDL_dynapi.sym
- #
- # Add before "extra symbols go here" line
- input = open(sdl_dynapi_sym)
- new_input = []
- for line in input:
- if "extra symbols go here" in line:
- new_input.append(" " + func_name + ";\n")
- new_input.append(line)
- input.close()
- f = open(sdl_dynapi_sym, 'w')
- for line in new_input:
- f.write(line)
- f.close()
- if __name__ == '__main__':
- parser = argparse.ArgumentParser()
- parser.add_argument('--dump', help='output all SDL API into a .json file', action='store_true')
- parser.add_argument('--debug', help='add debug traces', action='store_true')
- args = parser.parse_args()
- try:
- main()
- except Exception as e:
- print(e)
- exit(-1)
- print("done!")
- exit(0)
|