2018-09-27 03:14:16 +00:00
|
|
|
lovr = require 'lovr'
|
|
|
|
|
2018-04-26 03:07:28 +00:00
|
|
|
local function nogame()
|
2017-12-19 00:01:12 +00:00
|
|
|
function lovr.conf(t)
|
2020-09-25 22:41:30 +00:00
|
|
|
t.headset.supersample = true
|
2017-12-19 00:01:12 +00:00
|
|
|
t.modules.audio = false
|
|
|
|
t.modules.physics = false
|
2018-04-26 03:07:28 +00:00
|
|
|
t.modules.thread = false
|
2017-12-19 00:01:12 +00:00
|
|
|
end
|
|
|
|
|
2019-05-09 00:11:25 +00:00
|
|
|
local models = {}
|
|
|
|
|
2017-12-19 00:01:12 +00:00
|
|
|
function lovr.load()
|
2022-04-19 02:30:58 +00:00
|
|
|
print(string.format('LÖVR %d.%d.%d\nNo game', lovr.getVersion()))
|
|
|
|
lovr.event.quit()
|
2017-12-19 00:01:12 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
function lovr.draw()
|
2022-04-19 02:30:58 +00:00
|
|
|
--
|
2017-12-19 00:01:12 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-13 06:34:02 +00:00
|
|
|
-- Note: Cannot be overloaded
|
2018-04-26 03:07:28 +00:00
|
|
|
function lovr.boot()
|
|
|
|
local conf = {
|
2021-04-11 14:57:11 +00:00
|
|
|
version = '0.15.0',
|
2019-11-25 01:27:17 +00:00
|
|
|
identity = 'default',
|
2020-06-21 15:02:36 +00:00
|
|
|
saveprecedence = true,
|
2018-04-26 03:07:28 +00:00
|
|
|
modules = {
|
|
|
|
audio = true,
|
|
|
|
data = true,
|
|
|
|
event = true,
|
|
|
|
graphics = true,
|
|
|
|
headset = true,
|
|
|
|
math = true,
|
|
|
|
physics = true,
|
2021-02-25 16:00:12 +00:00
|
|
|
system = true,
|
2018-04-26 03:07:28 +00:00
|
|
|
thread = true,
|
|
|
|
timer = true
|
|
|
|
},
|
2021-02-05 11:48:18 +00:00
|
|
|
audio = {
|
|
|
|
start = true,
|
2021-03-08 05:42:17 +00:00
|
|
|
spatializer = nil
|
2021-02-05 11:48:18 +00:00
|
|
|
},
|
2022-04-27 05:48:51 +00:00
|
|
|
graphics = {
|
2022-05-11 22:28:04 +00:00
|
|
|
debug = false,
|
|
|
|
vsync = false
|
2022-04-27 05:48:51 +00:00
|
|
|
},
|
2018-04-26 03:07:28 +00:00
|
|
|
headset = {
|
2022-03-22 22:47:18 +00:00
|
|
|
drivers = { 'openxr', 'webxr', 'desktop' },
|
2020-09-25 22:41:30 +00:00
|
|
|
supersample = false,
|
2018-07-29 01:12:57 +00:00
|
|
|
offset = 1.7,
|
2020-11-18 00:38:27 +00:00
|
|
|
msaa = 4,
|
|
|
|
overlay = false
|
2018-04-26 03:07:28 +00:00
|
|
|
},
|
2019-06-10 07:11:20 +00:00
|
|
|
math = {
|
2019-07-01 09:16:13 +00:00
|
|
|
globals = true
|
2019-06-10 07:11:20 +00:00
|
|
|
},
|
2018-04-26 03:07:28 +00:00
|
|
|
window = {
|
2022-05-09 19:43:19 +00:00
|
|
|
width = 800,
|
2018-04-26 03:07:28 +00:00
|
|
|
height = 600,
|
|
|
|
fullscreen = false,
|
2019-08-28 07:56:46 +00:00
|
|
|
resizable = false,
|
2018-04-26 03:07:28 +00:00
|
|
|
title = 'LÖVR',
|
2022-05-09 19:43:19 +00:00
|
|
|
icon = nil
|
2018-04-26 03:07:28 +00:00
|
|
|
}
|
|
|
|
}
|
2017-12-19 00:01:12 +00:00
|
|
|
|
2018-04-26 03:07:28 +00:00
|
|
|
lovr.filesystem = require('lovr.filesystem')
|
|
|
|
local hasConf, hasMain = lovr.filesystem.isFile('conf.lua'), lovr.filesystem.isFile('main.lua')
|
|
|
|
if not lovr.filesystem.getSource() or not (hasConf or hasMain) then nogame() end
|
2017-12-19 00:01:12 +00:00
|
|
|
|
2022-03-28 20:24:44 +00:00
|
|
|
-- Shift args up in fused mode, instead of consuming one for the source path
|
|
|
|
if lovr.filesystem.isFused() then
|
|
|
|
for i = 1, #arg + 1 do
|
|
|
|
arg[i] = arg[i - 1]
|
|
|
|
end
|
|
|
|
arg[0] = lovr.filesystem.getSource()
|
|
|
|
end
|
|
|
|
|
2018-11-23 07:44:56 +00:00
|
|
|
local confOk, confError = true
|
|
|
|
if hasConf then confOk, confError = pcall(require, 'conf') end
|
|
|
|
if confOk and lovr.conf then confOk, confError = pcall(lovr.conf, conf) end
|
2018-04-26 03:07:28 +00:00
|
|
|
|
|
|
|
lovr._setConf(conf)
|
2020-06-21 15:02:36 +00:00
|
|
|
lovr.filesystem.setIdentity(conf.identity, conf.saveprecedence)
|
2017-12-19 00:01:12 +00:00
|
|
|
|
2018-11-16 10:27:34 +00:00
|
|
|
for module in pairs(conf.modules) do
|
2018-04-26 03:07:28 +00:00
|
|
|
if conf.modules[module] then
|
2018-09-27 18:45:43 +00:00
|
|
|
local ok, result = pcall(require, 'lovr.' .. module)
|
2018-11-16 10:27:34 +00:00
|
|
|
if not ok then
|
|
|
|
print(string.format('Warning: Could not load module %q: %s', module, result))
|
|
|
|
else
|
|
|
|
lovr[module] = result
|
|
|
|
end
|
2018-04-26 03:07:28 +00:00
|
|
|
end
|
2017-12-19 00:01:12 +00:00
|
|
|
end
|
|
|
|
|
2022-05-09 19:43:19 +00:00
|
|
|
if lovr.system and conf.window then
|
|
|
|
lovr.system.openWindow(conf.window)
|
|
|
|
end
|
|
|
|
|
2022-04-21 07:27:13 +00:00
|
|
|
if lovr.graphics then
|
|
|
|
lovr.graphics.init()
|
|
|
|
end
|
|
|
|
|
2020-08-25 21:49:19 +00:00
|
|
|
if lovr.headset and lovr.graphics and conf.window then
|
2021-06-10 23:26:15 +00:00
|
|
|
lovr.headset.start()
|
2020-08-25 21:49:19 +00:00
|
|
|
end
|
|
|
|
|
2018-04-26 03:07:28 +00:00
|
|
|
lovr.handlers = setmetatable({}, { __index = lovr })
|
2018-11-23 07:44:56 +00:00
|
|
|
if not confOk then error(confError) end
|
2018-04-26 03:07:28 +00:00
|
|
|
if hasMain then require 'main' end
|
|
|
|
return lovr.run()
|
2017-12-19 00:01:12 +00:00
|
|
|
end
|
|
|
|
|
2018-04-26 03:07:28 +00:00
|
|
|
function lovr.run()
|
2021-02-09 15:26:44 +00:00
|
|
|
if lovr.timer then lovr.timer.step() end
|
2018-04-26 03:07:28 +00:00
|
|
|
if lovr.load then lovr.load(arg) end
|
|
|
|
return function()
|
2021-02-09 15:26:44 +00:00
|
|
|
if lovr.event then
|
|
|
|
lovr.event.pump()
|
|
|
|
for name, a, b, c, d in lovr.event.poll() do
|
|
|
|
if name == 'restart' then
|
|
|
|
local cookie = lovr.restart and lovr.restart()
|
|
|
|
return 'restart', cookie
|
|
|
|
elseif name == 'quit' and (not lovr.quit or not lovr.quit(a)) then
|
|
|
|
return a or 0
|
|
|
|
end
|
|
|
|
if lovr.handlers[name] then lovr.handlers[name](a, b, c, d) end
|
2018-04-26 03:07:28 +00:00
|
|
|
end
|
2017-12-19 00:01:12 +00:00
|
|
|
end
|
2022-03-23 00:52:16 +00:00
|
|
|
local dt = 0
|
2021-02-09 15:26:44 +00:00
|
|
|
if lovr.timer then dt = lovr.timer.step() end
|
2022-03-23 00:52:16 +00:00
|
|
|
if lovr.headset then dt = lovr.headset.update() end
|
2018-04-26 03:07:28 +00:00
|
|
|
if lovr.update then lovr.update(dt) end
|
2022-05-30 22:35:07 +00:00
|
|
|
if lovr.graphics then
|
2022-06-06 03:38:14 +00:00
|
|
|
local headset = lovr.headset and lovr.headset.getTexture()
|
|
|
|
if headset then
|
|
|
|
local pass = lovr.graphics.getPass('render', headset)
|
|
|
|
for i = 1, lovr.headset.getViewCount() do
|
|
|
|
pass:setViewPose(i, lovr.headset.getViewPose(i))
|
|
|
|
pass:setProjection(i, lovr.headset.getViewAngles(i))
|
|
|
|
end
|
2022-06-09 06:59:48 +00:00
|
|
|
local skip = lovr.draw and lovr.draw(pass)
|
|
|
|
if not skip then lovr.graphics.submit(pass) end
|
2022-06-06 03:38:14 +00:00
|
|
|
end
|
|
|
|
if lovr.system.isWindowOpen() then
|
|
|
|
lovr.mirror(pass)
|
|
|
|
end
|
2022-05-30 22:35:07 +00:00
|
|
|
end
|
2022-06-06 03:38:14 +00:00
|
|
|
if lovr.headset then lovr.headset.submit() end
|
2021-02-09 15:26:44 +00:00
|
|
|
if lovr.math then lovr.math.drain() end
|
2017-12-19 00:01:12 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-06-06 03:38:14 +00:00
|
|
|
function lovr.mirror(pass)
|
|
|
|
if lovr.headset then
|
|
|
|
--
|
|
|
|
else
|
|
|
|
local pass = lovr.graphics.getPass('render', 'window')
|
2022-06-09 06:59:48 +00:00
|
|
|
local skip = lovr.draw and lovr.draw(pass)
|
|
|
|
if not skip then lovr.graphics.submit(pass) end
|
2022-06-06 03:38:14 +00:00
|
|
|
end
|
2019-01-24 21:20:14 +00:00
|
|
|
end
|
|
|
|
|
2018-11-13 06:34:02 +00:00
|
|
|
local function formatTraceback(s)
|
|
|
|
return s:gsub('\n[^\n]+$', ''):gsub('\t', ''):gsub('stack traceback', '\nStack')
|
|
|
|
end
|
|
|
|
|
|
|
|
function lovr.errhand(message, traceback)
|
2018-11-23 07:44:56 +00:00
|
|
|
message = tostring(message)
|
2019-08-01 07:14:37 +00:00
|
|
|
message = message .. formatTraceback(traceback or debug.traceback('', 4))
|
|
|
|
print('Error:\n' .. message)
|
2022-04-19 02:30:58 +00:00
|
|
|
return function() return 1 end
|
2017-12-19 00:01:12 +00:00
|
|
|
end
|
|
|
|
|
2018-03-12 17:37:45 +00:00
|
|
|
function lovr.threaderror(thread, err)
|
2018-04-26 03:07:28 +00:00
|
|
|
error('Thread error\n\n' .. err, 0)
|
2018-03-12 17:37:45 +00:00
|
|
|
end
|
|
|
|
|
lovr.log;
lovr.log is a new callback that is invoked whenever LÖVR wants to
send the project a message. For example, this could be a performance
warning from the graphics module, an error message from one of the
headset backends, or an API deprecation notice.
The callback's signature is (message, level, tag). The message is a
string containing the message to log, level is a string that is currently
one of "debug", "info", "warn", "error", and tag is an optional string
that is used to indicate the source of the message for grouping purposes.
The default implementation of the callback just prints the message,
but the callback can be overridden to do things like filter messages,
write them to a file, or even render them in VR. Projects can also
invoke the callback directly to log their own messages.
2020-07-06 22:20:55 +00:00
|
|
|
function lovr.log(message, level, tag)
|
2020-07-29 20:24:57 +00:00
|
|
|
message = message:gsub('\n$', '')
|
lovr.log;
lovr.log is a new callback that is invoked whenever LÖVR wants to
send the project a message. For example, this could be a performance
warning from the graphics module, an error message from one of the
headset backends, or an API deprecation notice.
The callback's signature is (message, level, tag). The message is a
string containing the message to log, level is a string that is currently
one of "debug", "info", "warn", "error", and tag is an optional string
that is used to indicate the source of the message for grouping purposes.
The default implementation of the callback just prints the message,
but the callback can be overridden to do things like filter messages,
write them to a file, or even render them in VR. Projects can also
invoke the callback directly to log their own messages.
2020-07-06 22:20:55 +00:00
|
|
|
print(message)
|
|
|
|
end
|
|
|
|
|
2018-11-13 06:34:02 +00:00
|
|
|
-- This splits up the string returned by luax_getstack so it looks like the error message plus the string from
|
|
|
|
-- debug.traceback(). This includes splitting on the newline before 'stack traceback:' and appending a newline
|
|
|
|
local function splitOnLabelLine(s, t)
|
|
|
|
local at = s:reverse():find(t:reverse())
|
|
|
|
if at then
|
|
|
|
local slen = #s
|
|
|
|
at = (#s - at - #t + 2)
|
|
|
|
return s:sub(1, at-2), s:sub(at,slen) .. '\n'
|
|
|
|
else
|
|
|
|
return s, ''
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- lovr will run this function in its own coroutine
|
2018-04-26 03:07:28 +00:00
|
|
|
return function()
|
2018-11-13 06:34:02 +00:00
|
|
|
local errored = false -- lovr.errhand may only be called once
|
|
|
|
local function onerror(e, tb) -- wrapper for errhand to ensure it is only called once
|
|
|
|
local function abortclean()
|
|
|
|
return 1
|
|
|
|
end
|
|
|
|
if not errored then
|
|
|
|
errored = true
|
|
|
|
return lovr.errhand(e, tb) or abortclean
|
|
|
|
else
|
2018-11-14 03:56:29 +00:00
|
|
|
print('Error occurred while trying to display another error:\n' ..
|
2018-11-23 07:44:56 +00:00
|
|
|
tostring(e) .. formatTraceback(tb or debug.traceback('', 2)))
|
2018-11-13 06:34:02 +00:00
|
|
|
return abortclean
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Executes lovr.boot and lovr.run.
|
|
|
|
-- continuation, afterward, will be either lovr.run's per-frame function, or the result of errhand.
|
|
|
|
local _, continuation = xpcall(lovr.boot, onerror)
|
2018-04-26 03:07:28 +00:00
|
|
|
|
|
|
|
while true do
|
2018-11-13 06:34:02 +00:00
|
|
|
if type(continuation) == 'string' then -- LuaJIT returns a fixed string if an error occurs in an xpcall error handler
|
2019-08-03 03:29:28 +00:00
|
|
|
print('Error occurred while trying to display another error: ' .. continuation)
|
2018-11-13 06:34:02 +00:00
|
|
|
return 1
|
|
|
|
end
|
2018-04-26 03:07:28 +00:00
|
|
|
|
2020-02-26 22:40:40 +00:00
|
|
|
local ok, result, extra = xpcall(continuation, onerror)
|
|
|
|
if result and ok then return result, extra -- Result is value returned by function. Return it.
|
2018-11-23 07:44:56 +00:00
|
|
|
elseif not ok then continuation = result end -- Result is value returned by error handler. Make it the new error handler.
|
|
|
|
|
2018-11-13 06:34:02 +00:00
|
|
|
local externerror = coroutine.yield() -- Return control to C code
|
|
|
|
|
|
|
|
if externerror then -- A must-report error occurred in the C code
|
|
|
|
local errorpart, tracepart = splitOnLabelLine(externerror, 'stack traceback:')
|
|
|
|
continuation = onerror(errorpart, tracepart) -- Switch continuation to lovr.errhand
|
|
|
|
end
|
|
|
|
end
|
2017-12-19 00:01:12 +00:00
|
|
|
end
|