mirror of
				https://github.com/encounter/SDL.git
				synced 2025-10-25 03:00:23 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			462 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Lua
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			462 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Lua
		
	
	
		
			Executable File
		
	
	
	
	
| -- Copyright (C) 1997-2016 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.
 | |
| --
 | |
| -- Meta-build system using premake created and maintained by
 | |
| -- Benjamin Henning <b.henning@digipen.edu>
 | |
| 
 | |
| --[[
 | |
| sdl_projects.lua
 | |
| 
 | |
| 	This file contains all the functions which are needed to define any project
 | |
| 	within the meta-build system. Many of these functions serve as
 | |
| 	pseudo-replacements for many similarly named premake functions, and that is
 | |
| 	intentional. Even the implementation of these functions are intended to look
 | |
| 	similar to regular premake code. These functions serve to dramatically
 | |
| 	simplify the project definition process to just a few lines of code, versus
 | |
| 	the many more needed for projects defined purely with premake.
 | |
| 
 | |
| 	This approach is possible because this meta-build system adds another layer of
 | |
| 	indirection to the premake system, creating a sort of 'meta-meta-build'
 | |
| 	system. Nevertheless, there is a lot more flexibility because the meta-build
 | |
| 	system itself can be used to check for dependencies in a much more complex way
 | |
| 	than premake originally intended. All of the functions useful to the project
 | |
| 	definition system are contained in this file and are documented.
 | |
| ]]
 | |
| 
 | |
| projects = { }
 | |
| 
 | |
| local currentProject = nil
 | |
| local currentDep = nil
 | |
| local nextFuncCompat = true -- by default, unless state otherwise changed
 | |
| local dependencyFunctions = { }
 | |
| local dependencyResults = { } -- for when the dependencies are executed
 | |
| 
 | |
| -- query whether this function is compatible; resets internal state of
 | |
| -- compatibility to true until SDL_isos is called again
 | |
| local function oscompat()
 | |
| 	local compat = nextFuncCompat
 | |
| 	nextFuncCompat = true
 | |
| 	return compat
 | |
| end
 | |
| 
 | |
| -- determine whether the specific OS name is within a pattern.
 | |
| local function osmatch(name, pattern)
 | |
| 	local checks = pattern:explode('|')
 | |
| 	for i,v in pairs(checks) do
 | |
| 		if name == v then
 | |
| 			return true
 | |
| 		end
 | |
| 	end
 | |
| 	return false
 | |
| end
 | |
| 
 | |
| -- Registers a dependency checker function based on a name. This function is
 | |
| -- used in order to determine compatibility with the current system for a given
 | |
| -- SDL_dependency. See SDL_depfunc for more information.
 | |
| --
 | |
| -- Specifies a function which will be invoked upon determining whether this
 | |
| -- dependency is valid for the current system setup (ie, whether the system
 | |
| -- has the right architecture, operating system, or even if it's installed).
 | |
| -- The dependency function takes no arguments, but it must return the following
 | |
| -- values:
 | |
| --
 | |
| --   <foundDep> [includePaths] [libPaths] [inputLibLibraries]
 | |
| --
 | |
| -- The last three are optional, unless foundDep is true. The name should be
 | |
| -- descriptive of the outside dependency, since it may be shown to the user.
 | |
| -- This function is intended to be used only after invoking SDL_dependency.
 | |
| function SDL_registerDependencyChecker(name, func)
 | |
| 	dependencyFunctions[name:lower()] = func
 | |
| end
 | |
| 
 | |
| -- Initializes the definition of a SDL project given the name of the project.
 | |
| function SDL_project(name)
 | |
| 	if not oscompat() then return end
 | |
| 	currentProject = { }
 | |
| 	currentProject.name = name
 | |
| 	currentProject.compat = true
 | |
| 	projects[name] = currentProject
 | |
| 	currentProject.dependencyTree = { }
 | |
| 	-- stores which dependencies have already been checked on behalf of this
 | |
| 	-- project
 | |
| 	currentProject.dependencyValues = { }
 | |
| 	currentDep = nil
 | |
| end
 | |
| 
 | |
| -- Specifies the build kind of the SDL project (e.g. StaticLib, SharedLib,
 | |
| -- ConsoleApp, etc.), based on premake presets.
 | |
| function SDL_kind(k)
 | |
| 	if not oscompat() then return end
 | |
| 	currentProject.kind = k
 | |
| end
 | |
| 
 | |
| -- Specifies which platforms this project supports. Note: this list is not the
 | |
| -- exact list of supported platforms in the generated project. The list of
 | |
| -- platforms this project supports will be the unique list of all combined
 | |
| -- projects for this SDL solution. Thus, only one project needs to actually
 | |
| -- maintain a list. This function is additive, that is, everytime it is called
 | |
| -- it adds it to a unique list of platforms
 | |
| function SDL_platforms(tbl)
 | |
| 	if not oscompat() then return end
 | |
| 	if not currentProject.platforms then
 | |
| 		currentProject.platforms = { }
 | |
| 	end
 | |
| 	for k,v in pairs(tbl) do
 | |
| 		currentProject.platforms[#currentProject.platforms + 1] = v
 | |
| 	end
 | |
| end
 | |
| 
 | |
| -- Specifies the programming language of the project, such as C or C++.
 | |
| function SDL_language(k)
 | |
| 	if not oscompat() then return end
 | |
| 	currentProject.language = k
 | |
| end
 | |
| 
 | |
| -- Specifies the root directory in which the meta-build system should search for
 | |
| -- source files, given the paths and files added.
 | |
| function SDL_sourcedir(src)
 | |
| 	if not oscompat() then return end
 | |
| 	currentProject.sourcedir = src
 | |
| end
 | |
| 
 | |
| -- Specifies the destination location of where the IDE files related to the
 | |
| -- project should be saved after generation.
 | |
| function SDL_projectLocation(loc)
 | |
| 	if not oscompat() then return end
 | |
| 	currentProject.projectLocation = loc
 | |
| end
 | |
| 
 | |
| -- Specifies a table of files that should be copied from the source directory
 | |
| -- to the end result build directory of the binary file.
 | |
| function SDL_copy(tbl)
 | |
| 	if not oscompat() then return end
 | |
| 	currentProject.copy = tbl
 | |
| end
 | |
| 
 | |
| -- Specifies a list of other SDL projects in this workspace the currently active
 | |
| -- project is dependent on. If the dependent project is a library, the binary
 | |
| -- result will be copied from its directory to the build directory of the
 | |
| -- currently active project automatically.
 | |
| function SDL_projectDependencies(tbl)
 | |
| 	if not oscompat() then return end
 | |
| 	currentProject.projectDependencies = tbl
 | |
| end
 | |
| 
 | |
| -- Specifies a list of compiler-level preprocessor definitions that should be
 | |
| -- set in the resulting project upon compile time. This adds to the current
 | |
| -- table of defines.
 | |
| function SDL_defines(tbl)
 | |
| 	if not oscompat() then return end
 | |
| 	if not currentProject.defines then
 | |
| 		currentProject.defines = { }
 | |
| 	end
 | |
| 	for k,v in pairs(tbl) do
 | |
| 		currentProject.defines[#currentProject.defines + 1] = v
 | |
| 	end
 | |
| end
 | |
| 
 | |
| -- Initializes an outside dependency this project has, such as X11 or DirectX.
 | |
| -- This function, once invoked, may change the behavior of other SDL
 | |
| -- project-related functions, so be sure to be familiar with all the functions
 | |
| -- and any specified behavior when used around SDL_dependency.
 | |
| function SDL_dependency(name)
 | |
| 	if not oscompat() then return end
 | |
| 	currentDep = { nil, compat = true, }
 | |
| 	currentDep.name = name
 | |
| 	table.insert(currentProject.dependencyTree, currentDep)
 | |
| end
 | |
| 
 | |
| -- Special function for getting the current OS. This factors in whether the
 | |
| -- metabuild system is in MinGW, Cygwin, or iOS mode.
 | |
| function SDL_getos()
 | |
| 	if _OPTIONS["ios"] ~= nil then
 | |
| 		return "ios"
 | |
| 	elseif _OPTIONS["mingw"] ~= nil then
 | |
| 		return "mingw"
 | |
| 	elseif _OPTIONS["cygwin"] ~= nil then
 | |
| 		return "cygwin"
 | |
| 	end
 | |
| 	return os.get()
 | |
| end
 | |
| 
 | |
| -- Specifies which operating system this dependency targets, such as windows or
 | |
| -- macosx, as per premake presets.
 | |
| function SDL_os(name)
 | |
| 	if not oscompat() then return end
 | |
| 	if not currentProject then return end
 | |
| 	if not currentDep then
 | |
| 		currentProject.opsys = name
 | |
| 		currentProject.compat = osmatch(SDL_getos(), name)
 | |
| 	else
 | |
| 		currentDep.opsys = name
 | |
| 		currentDep.compat = osmatch(SDL_getos(), name)
 | |
| 	end
 | |
| end
 | |
| 
 | |
| -- Specifies which operating system this dependency does not targets. This is
 | |
| -- for nearly platform-independent projects or dependencies that will not work
 | |
| -- on specific systems, such as ios.
 | |
| function SDL_notos(name)
 | |
| 	if not oscompat() then return end
 | |
| 	if not currentProject then return end
 | |
| 	if not currentDep then
 | |
| 		currentProject.opsys = "~" .. name
 | |
| 		currentProject.compat = not osmatch(SDL_getos(), name)
 | |
| 	else
 | |
| 		currentDep.opsys = "~" .. name
 | |
| 		currentDep.compat = not osmatch(SDL_getos(), name)
 | |
| 	end
 | |
| end
 | |
| 
 | |
| -- Changes the internal state of function compatibility based on whether the
 | |
| -- current os is the one expected; the next function will be affected by this
 | |
| -- change, but no others. The name can be a pattern using '|' to separate
 | |
| -- multiple operating systems, such as:
 | |
| --   SDL_isos("windows|macosx")
 | |
| function SDL_isos(name)
 | |
| 	nextFuncCompat = osmatch(SDL_getos(), name)
 | |
| end
 | |
| 
 | |
| -- Same as SDL_isos, except it negates the internal state for exclusion
 | |
| -- checking.
 | |
| function SDL_isnotos(name)
 | |
| 	nextFuncCompat = not osmatch(SDL_getos(), name)
 | |
| end
 | |
| 
 | |
| -- Changes the internal state of function compatibility based on whether the
 | |
| -- current system is running a 64bit Operating System and architecture; the
 | |
| -- next function will be affected by this change, but none thereafter.
 | |
| function SDL_is64bit()
 | |
| 	nextFuncCompat = os.is64bit()
 | |
| end
 | |
| 
 | |
| -- Same as SDL_is64bit, except it negates the internal state for
 | |
| -- exclusion checking.
 | |
| function SDL_isnot64bit()
 | |
| 	nextFuncCompat = not os.is64bit()
 | |
| end
 | |
| 
 | |
| -- Look at SDL_depfunc and SDL_notdepfunc for detailed information about this
 | |
| -- function.
 | |
| local function SDL_depfunc0(funcname, exclude)
 | |
| 	if not oscompat() then return end
 | |
| 	if not currentDep.compat then return end
 | |
| 	local force = _OPTIONS[funcname:lower()] ~= nil
 | |
| 	local func = dependencyFunctions[funcname:lower()]
 | |
| 	if not func then
 | |
| 		print("Warning: could not find dependency function named: " .. funcname)
 | |
| 		currentDep.compat = false
 | |
| 		return
 | |
| 	end
 | |
| 	local cachedFuncResults = dependencyResults[funcname:lower()]
 | |
| 	local depFound, depInc, depLib, depInput
 | |
| 	if cachedFuncResults then
 | |
| 		depFound = cachedFuncResults.depFound
 | |
| 		-- just skip the rest of the function, the user was already warned
 | |
| 		-- exclude mode varies the compatibility slightly
 | |
| 		if force then
 | |
| 			depFound = true
 | |
| 		end
 | |
| 		if not depFound and not exclude then
 | |
| 			currentDep.compat = false
 | |
| 			return
 | |
| 		elseif depFound and exclude then
 | |
| 			currentDep.compat = false
 | |
| 			return
 | |
| 		end
 | |
| 		depInc = cachedFuncResults.depInc
 | |
| 		depLib = cachedFuncResults.depLib
 | |
| 		depInput = cachedFuncResults.depInput
 | |
| 	else
 | |
| 		local result = func()
 | |
| 		if result.found then
 | |
| 			depFound = result.found
 | |
| 		else
 | |
| 			depFound = false
 | |
| 		end
 | |
| 		if force then
 | |
| 			depFound = true
 | |
| 		end
 | |
| 		if result.incDirs then
 | |
| 			depInc = result.incDirs
 | |
| 		else
 | |
| 			depInc = { }
 | |
| 		end
 | |
| 		if result.libDirs then
 | |
| 			depLib = result.libDirs
 | |
| 		else
 | |
| 			depLib = { }
 | |
| 		end
 | |
| 		if result.libs then
 | |
| 			depInput = result.libs
 | |
| 		else
 | |
| 			depInput = { }
 | |
| 		end
 | |
| 		cachedFuncResults = { }
 | |
| 		cachedFuncResults.depFound = depFound
 | |
| 		cachedFuncResults.depInc = depInc
 | |
| 		cachedFuncResults.depLib = depLib
 | |
| 		cachedFuncResults.depInput = depInput
 | |
| 		dependencyResults[funcname:lower()] = cachedFuncResults
 | |
| 		if not depFound and not exclude then
 | |
| 			currentDep.compat = false
 | |
| 			return
 | |
| 		elseif depFound and exclude then
 | |
| 			currentDep.compat = false
 | |
| 			return
 | |
| 		end
 | |
| 	end
 | |
| 	-- we only want to embed this dependency if we're not in exclude mode
 | |
| 	if depFound and not exclude then
 | |
| 		local dependency = { }
 | |
| 		if not currentDep.includes then
 | |
| 			currentDep.includes = { }
 | |
| 		end
 | |
| 		for k,v in pairs(depInc) do
 | |
| 			currentDep.includes[v] = v
 | |
| 		end
 | |
| 		if not currentDep.libs then
 | |
| 			currentDep.libs = { }
 | |
| 		end
 | |
| 		for k,v in pairs(depLib) do
 | |
| 			currentDep.libs[v] = v
 | |
| 		end
 | |
| 		if not currentDep.links then
 | |
| 			currentDep.links = { }
 | |
| 		end
 | |
| 		for k,v in pairs(depInput) do
 | |
| 			currentDep.links[v] = v
 | |
| 		end
 | |
| 	else -- end of dependency found check
 | |
| 		-- if we are not excluding this dependency, then print a warning
 | |
| 		-- if not found
 | |
| 		if not exclude then
 | |
| 			print("Warning: could not find dependency: " .. funcname)
 | |
| 		end
 | |
| 		currentDep.compat = exclude
 | |
| 	end
 | |
| end
 | |
| 
 | |
| -- Given a dependency name, this function will register the dependency and try
 | |
| -- to pair it with a dependency function that was registered through
 | |
| -- SDL_registerDependencyChecker. If the function is not found, compatibility
 | |
| -- will automatically be dropped for this project and a warning will be printed
 | |
| -- to the standard output. Otherwise, the dependency function will be invoked
 | |
| -- and compatibility for the project will be updated. If the project currently
 | |
| -- is not compatible based on the Operating System or previous dependency, the
 | |
| -- dependency function will not be checked at all and this function will
 | |
| -- silently return.
 | |
| function SDL_depfunc(funcname)
 | |
| 	SDL_depfunc0(funcname, false)
 | |
| end
 | |
| 
 | |
| -- Same as SDL_depfunc, except this forces dependency on the function failing,
 | |
| -- rather than succeeding. This is useful for situations where two different
 | |
| -- files are required based on whether a dependency is found (such as the
 | |
| -- joystick and haptic systems).
 | |
| function SDL_notdepfunc(funcname)
 | |
| 	SDL_depfunc0(funcname, true)
 | |
| end
 | |
| 
 | |
| -- Determines whether the specified dependency is supported without actually
 | |
| -- executing the dependency or changing the internal states of the current
 | |
| -- project or dependency definition. This function will only work if the
 | |
| -- dependency has already been checked and its results cached within the
 | |
| -- definition system. This function returns true if the dependency is known to
 | |
| -- be supported, or false if otherwise (or if it cannot be known at this time).
 | |
| function SDL_assertdepfunc(funcname)
 | |
| 	-- if forced, then of course it's on
 | |
| 	if _OPTIONS[funcname:lower()] then
 | |
| 		return true
 | |
| 	end
 | |
| 	local results = dependencyResults[funcname:lower()]
 | |
| 	if not results or not results.depFound then
 | |
| 		-- either not excuted yet, doesn't exist, or wasn't found
 | |
| 		print("Warning: required dependency not found: " .. funcname ..
 | |
| 			". Make sure your dependencies are in a logical order.")
 | |
| 		return false
 | |
| 	end
 | |
| 	return true
 | |
| end
 | |
| 
 | |
| -- Returns a list of currently registered dependencies. The values within the
 | |
| -- table will be sorted, but their names will be lowercased due to internal
 | |
| -- handling of case-insensitive dependency names.
 | |
| function SDL_getDependencies()
 | |
| 	local deps = { }
 | |
| 	for k,_ in pairs(dependencyFunctions) do
 | |
| 		deps[#deps + 1] = k
 | |
| 	end
 | |
| 	table.sort(deps)
 | |
| 	return deps
 | |
| end
 | |
| 
 | |
| -- Specifies a list of libraries that should always be linked to in this
 | |
| -- project, regardless of a dependency function. If after a dependency
 | |
| -- declaration, these files will only be included in the project if the
 | |
| -- dependency is compatible with the native system, given SDL_os usage and any
 | |
| -- sort of custom dependency function.
 | |
| function SDL_links(tbl)
 | |
| 	if not oscompat() then return end
 | |
| 	if currentDep and not currentDep.compat then return end
 | |
| 	if currentProject.customLinks == nil then
 | |
| 		currentProject.customLinks = { }
 | |
| 	end
 | |
| 	for i,v in ipairs(tbl) do
 | |
| 		currentProject.customLinks[#currentProject.customLinks + 1] = v
 | |
| 	end
 | |
| end
 | |
| 
 | |
| -- Specifies a list of configuration values that are assigned as preprocessor
 | |
| -- definitions in the SDL configuration header, used to globally configure
 | |
| -- features during the building of the SDL library. If after a dependency
 | |
| -- declaration, these files will only be included in the project if the
 | |
| -- dependency is compatible with the native system, given SDL_os usage and any
 | |
| -- sort of custom dependency function.
 | |
| function SDL_config(tbl)
 | |
| 	if not oscompat() then return end
 | |
| 	if not currentDep then
 | |
| 		currentProject.config = tbl
 | |
| 		return
 | |
| 	end
 | |
| 	if not currentDep.compat then return end
 | |
| 	currentDep.config = tbl
 | |
| end
 | |
| 
 | |
| -- Specifies a list of paths where all .c, .h, and .m files should be included
 | |
| -- for compiling, where the source directory is the root. If after a dependency
 | |
| -- declaration, these files will only be included in the project if the
 | |
| -- dependency is compatible with the native system, given SDL_os usage and any
 | |
| -- sort of custom dependency function.
 | |
| function SDL_paths(tbl)
 | |
| 	if not oscompat() then return end
 | |
| 	if not currentDep then
 | |
| 		currentProject.paths = tbl
 | |
| 		return
 | |
| 	end
 | |
| 	if not currentDep.compat then return end
 | |
| 	currentDep.paths = tbl
 | |
| end
 | |
| 
 | |
| -- Specifies a list of files found within the source directory that this project
 | |
| -- should include during compile time. If after a dependency declaration, these
 | |
| -- files will only be included in the project if the dependency is compatible
 | |
| -- with the native system, given SDL_os usage and any sort of custom dependency
 | |
| -- function.
 | |
| function SDL_files(tbl)
 | |
| 	if not oscompat() then return end
 | |
| 	if not currentDep then
 | |
| 		currentProject.files = tbl
 | |
| 		return
 | |
| 	end
 | |
| 	if not currentDep.compat then return end
 | |
| 	currentDep.files = tbl
 | |
| end
 |