diff --git a/source/duke3d/src/duke3d.h b/source/duke3d/src/duke3d.h index 6035ebd1d..14af83e42 100644 --- a/source/duke3d/src/duke3d.h +++ b/source/duke3d/src/duke3d.h @@ -150,26 +150,14 @@ EDUKE32_STATIC_ASSERT(7 <= MAXTILES-MAXUSERTILES); #include "sounds.h" #include "soundsdyn.h" -#ifdef LUNATIC -# include "lunatic_game.h" -#endif - static inline int32_t G_HaveActor(int spriteNum) { -#ifdef LUNATIC - return El_HaveActor(spriteNum); -#else return g_tile[spriteNum].execPtr!=NULL; -#endif } static inline int32_t G_DefaultActorHealth(int spriteNum) { -#ifdef LUNATIC - return g_elActors[spriteNum].strength; -#else return G_HaveActor(spriteNum) ? g_tile[spriteNum].execPtr[0] : 0; -#endif } #endif diff --git a/source/duke3d/src/gameexec.cpp b/source/duke3d/src/gameexec.cpp index d9ecd6cd3..898514a1f 100644 --- a/source/duke3d/src/gameexec.cpp +++ b/source/duke3d/src/gameexec.cpp @@ -32,10 +32,6 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. #include "savegame.h" #include "scriplib.h" -#ifdef LUNATIC -# include "lunatic_game.h" -#endif - #include "vfs.h" #if KRANDDEBUG diff --git a/source/duke3d/src/gameexec.h b/source/duke3d/src/gameexec.h index a227311e8..72926cb42 100644 --- a/source/duke3d/src/gameexec.h +++ b/source/duke3d/src/gameexec.h @@ -28,10 +28,6 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. #include "gamedef.h" // vmstate_t #include "sector.h" // mapstate_t -#ifdef LUNATIC -# include "lunatic_game.h" -#endif - int32_t VM_ExecuteEvent(int const nEventID, int const spriteNum, int const playerNum, int const nDist, int32_t const nReturn); int32_t VM_ExecuteEvent(int const nEventID, int const spriteNum, int const playerNum, int const nDist); int32_t VM_ExecuteEvent(int const nEventID, int const spriteNum, int const playerNum); @@ -39,11 +35,7 @@ int32_t VM_ExecuteEventWithValue(int const nEventID, int const spriteNum, int co static FORCE_INLINE bool VM_HaveEvent(int const nEventID) { -#ifdef LUNATIC - return L_IsInitialized(&g_ElState) && El_HaveEvent(nEventID); -#else return !!apScriptEvents[nEventID]; -#endif } static FORCE_INLINE int32_t VM_OnEvent(int nEventID, int spriteNum, int playerNum, int nDist, int32_t nReturn) diff --git a/source/duke3d/src/lunatic/_defs_editor.lua b/source/duke3d/src/lunatic/_defs_editor.lua deleted file mode 100644 index 399db6432..000000000 --- a/source/duke3d/src/lunatic/_defs_editor.lua +++ /dev/null @@ -1,55 +0,0 @@ --- INTERNAL --- definitions of BUILD and game types for the Lunatic Interpreter - -local ffi = require("ffi") -local ffiC = ffi.C - -ffi.cdef[[ -enum { - LUNATIC_CLIENT_MAPSTER32 = 0, - LUNATIC_CLIENT_EDUKE32 = 1, - - LUNATIC_CLIENT = LUNATIC_CLIENT_MAPSTER32 -} -]] - ---== First, load the definitions common to the game's and editor's Lua interface. -decl = ffi.cdef -local defs_c = require("defs_common") -defs_c.finish_spritetype({}) - -defs_c.create_globals(_G) - --- TODO: provide access to only a subset, restict access to ffiC? -gv = ffiC - ---== Mapster32-specific initialization - -ffi.cdef "char *listsearchpath(int32_t initp);" - --- Add the search path directories to the Lua load path. -local initp = 1 -while (true) do - local sp_c = ffiC.listsearchpath(initp) - - if (sp_c == nil) then - break - end - - local sp = ffi.string(sp_c) - assert(sp:sub(-1)=='/') - package.path = sp..'?.lua;'..package.path - - initp = 0 -end - --- Helper functions -local package = package -local require = require - -function reload(modname) - package.loaded[modname] = nil - return require(modname) -end - ---print('Lua load path: '..package.path) diff --git a/source/duke3d/src/lunatic/_defs_game.lua b/source/duke3d/src/lunatic/_defs_game.lua deleted file mode 100644 index b8cc22626..000000000 --- a/source/duke3d/src/lunatic/_defs_game.lua +++ /dev/null @@ -1,2632 +0,0 @@ --- INTERNAL --- definitions of BUILD and game types for the Lunatic Interpreter - -local require = require -local ffi = require("ffi") -local ffiC = ffi.C - --- Lua C API functions. -local CF = CF - -local bit = bit -local coroutine = coroutine -local string = string -local table = table -local math = math - -local assert = assert -local error = error -local getfenv = getfenv -local getmetatable = getmetatable -local ipairs = ipairs -local loadstring = loadstring -local pairs = pairs -local pcall = pcall -local rawget = rawget -local rawset = rawset -local select = select -local setmetatable = setmetatable -local setfenv = setfenv -local tonumber = tonumber -local tostring = tostring -local type = type - --- Create a new module for passing stuff to other modules. -local lprivate = {} -require("package").loaded.lprivate = lprivate - -require("jit.opt").start("maxmcode=10240") -- in KiB - --- The "gv" global will provide access to C global *scalars* and safe functions. --- NOTE: This exposes C library functions from e.g. the global C namespace, but --- without their declarations, they will be sitting there like a stone. -local gv_ = {} --- [key]= forbids, [key]= overrides -local gv_access = {} - --- This is for declarations of arrays or pointers which should not be --- accessible through the "gv" global. The "defs_common" module will --- use this function. --- --- Notes: do not declare multiple scalars on one line (this is bad: --- "int32_t a, b"). Do not name array arguments (or add a space --- between the identifier and the '[' instead). -function decl(str, ...) - -- NOTE that the regexp also catches non-array/non-function identifiers - -- like "user_defs ud;" - for varname in string.gmatch(str, "([%a_][%w_]*)[[(;]") do - if (ffiC._DEBUG_LUNATIC ~= 0) then - print("FORBID "..varname) - end - gv_access[varname] = true - end - - ffi.cdef(str, ...) -end - -lprivate.decl = decl - -ffi.cdef[[ -enum { - LUNATIC_CLIENT_MAPSTER32 = 0, - LUNATIC_CLIENT_EDUKE32 = 1, - - LUNATIC_CLIENT = LUNATIC_CLIENT_EDUKE32 -} -]] - --- Load the definitions common to the game's and editor's Lua interface. -local defs_c = require("defs_common") -local cansee = defs_c.cansee -local strip_const = defs_c.strip_const -local setmtonce = defs_c.setmtonce - --- Must be after loading "defs_common" which redefines "print" to use --- OSD_Printf() -local print, printf = print, defs_c.printf - - ----=== EDuke32 game definitions ===--- - -local INV_NAMES = { - "STEROIDS", - "SHIELD", - "SCUBA", - "HOLODUKE", - "JETPACK", - "DUMMY1", - "ACCESS", - "HEATS", - "DUMMY2", - "FIRSTAID", - "BOOTS", -} - -local WEAPON_NAMES = { - "KNEE", - "PISTOL", - "SHOTGUN", - "CHAINGUN", - "RPG", - "HANDBOMB", - "SHRINKER", - "DEVISTATOR", - "TRIPBOMB", - "FREEZE", - "HANDREMOTE", - "GROW", -} - ----- game structs ---- - -lprivate.GET = defs_c.conststruct(INV_NAMES) -lprivate.WEAPON = defs_c.conststruct(WEAPON_NAMES) - -ffi.cdef([[ -enum { - GET_MAX = 11, - MAX_WEAPONS = 12, - MAXPLAYERS = 16, - GTICSPERSEC = 30, // The real number of movement updates per second -}; -]]) - -ffi.cdef[[ -struct action { - int16_t startframe, numframes; - int16_t viewtype, incval, delay; - uint16_t flags; -}; - -struct move { - int16_t hvel, vvel; -}; - -#pragma pack(push,1) -typedef struct { int32_t id; struct move mv; } con_move_t; -typedef struct { int32_t id; struct action ac; } con_action_t; -#pragma pack(pop) - -typedef struct { - int32_t id; - con_action_t act; - con_move_t mov; - int32_t movflags; -} con_ai_t; -]] - -defs_c.bitint_new_struct_type("int16_t", "SBit16") -defs_c.bitint_new_struct_type("int32_t", "SBit32") -defs_c.bitint_new_struct_type("uint32_t", "UBit32") - --- Struct template for actor_t. It already has 'const' fields (TODO: might need --- to make more 'const'), but still has array members exposed, so is unsuited --- for external exposure. -local ACTOR_STRUCT = [[ -struct { - const int32_t t_data[10]; - const struct move mv; - const struct action ac; - const uint16_t actiontics; - -]]..defs_c.bitint_member("SBit32", "flags")..[[ - vec3_t bpos; //12b - int32_t floorz,ceilingz,lastvx,lastvy; //16b - int32_t lasttransport; //4b - - const int16_t picnum; - int16_t ang, extra; - const int16_t owner; - // NOTE: not to be confused with .movflags: -]]..defs_c.bitint_member("SBit16", "_movflag")..[[ - int16_t tempang, timetosleep; - - int16_t stayputsect; - const int16_t dispicnum; - // Movement flags, sprite[i].hitag in C-CON. - // XXX: more research where it was used in EDuke32's C code? (also .lotag <-> actiontics) - // XXX: what if CON code knew of the above implementation detail? -]]..defs_c.bitint_member("UBit16", "movflags")..[[ - int16_t cgg; - - const int16_t lightId, lightcount, lightmaxrange; - // NOTE: on 32-bit, C's lightptr+filler <=> this dummy: - const union { intptr_t ptr; uint64_t dummy; } _light; -} -]] - -local bcarray = require("bcarray") - -local bcheck = require("bcheck") -local check_sector_idx, check_tile_idx = bcheck.sector_idx, bcheck.tile_idx -local check_sprite_idx = bcheck.sprite_idx -local check_weapon_idx, check_inventory_idx = bcheck.weapon_idx, bcheck.inventory_idx -local check_sound_idx = bcheck.sound_idx -local check_number = bcheck.number -local check_type = bcheck.type - -bcarray.new("int16_t", 64, "loogie", "int16_x_64") -- TODO: randomize member names -bcarray.new("int16_t", ffiC.MAX_WEAPONS, "weapon", "int16_x_MAX_WEAPONS", WEAPON_NAMES) -bcarray.new("int16_t", ffiC.GET_MAX, "inventory", "int16_x_GET_MAX", INV_NAMES) - --- NOTE: writing e.g. "ps.jetpack_on" in Lua when "ps.jetpack_on~=0" was meant --- is probably one of the most commonly committed errors, so we make it a bool --- type instead of uint8_t. The only issue is that if CON coders used these --- fields to store more than just one bit, we're in trouble. --- This will need to be documented and frozen for release. -local DUKEPLAYER_STRUCT = [[ -__attribute__((packed)) struct { - vec3_t pos, opos, vel, npos; - vec2_t bobpos, fric; - int32_t truefz, truecz, player_par; - int32_t randomflamex, exitx, exity; - int32_t runspeed, max_player_health, max_shield_amount; - int32_t autostep, autostep_sbw; - - uint32_t interface_toggle_flag; - - int32_t pipebombControl, pipebombLifetime, pipebombLifetimeVar; - int32_t tripbombControl, tripbombLifetime, tripbombLifetimeVar; - - int32_t zrange; - int16_t angrange, autoaimang; - - uint16_t max_actors_killed, actors_killed; -]]..defs_c.bitint_member("UBit16", "gotweapon")..[[ - uint16_t zoom; - - int16_x_64 loogiex; - int16_x_64 loogiey; - int16_t sbs, sound_pitch; - - int16_t ang, oang, angvel; - const int16_t cursectnum; - int16_t look_ang, last_extra, subweapon; - int16_x_MAX_WEAPONS max_ammo_amount; - int16_x_MAX_WEAPONS ammo_amount; - int16_x_GET_MAX inv_amount; - const int16_t wackedbyactor; - int16_t pyoff, opyoff; - - int16_t horiz, horizoff, ohoriz, ohorizoff; - const int16_t newowner; - int16_t jumping_counter, airleft; - int16_t fta; - const int16_t ftq; - const int16_t access_wallnum, access_spritenum; - int16_t got_access, weapon_ang, visibility; - int16_t somethingonplayer, on_crane; - const int16_t i; - const int16_t one_parallax_sectnum; - int16_t random_club_frame, one_eighty_count; - const int16_t dummyplayersprite; - int16_t extra_extra8; - int16_t actorsqu, timebeforeexit; - const int16_t customexitsound; - int16_t last_pissed_time; - - int16_x_MAX_WEAPONS weaprecs; - int16_t weapon_sway, crack_time, bobcounter; - - int16_t orotscrnang, rotscrnang, dead_flag; // JBF 20031220: added orotscrnang - int16_t holoduke_on, pycount; - int16_t transporter_hold; - - uint8_t max_secret_rooms, secret_rooms; - uint8_t frag, fraggedself, quick_kick, last_quick_kick; - uint8_t return_to_center; - bool reloading; - const uint8_t weapreccnt; - uint8_t aim_mode, auto_aim, weaponswitch, movement_lock, team; - uint8_t tipincs, hbomb_hold_delay; - const

uint8_t frag_ps; - uint8_t kickback_pic; - - uint8_t gm; - bool on_warping_sector; - uint8_t footprintcount, hurt_delay; - bool hbomb_on, jumping_toggle, rapid_fire_hold, on_ground; - // NOTE: there's array indexing with inven_icon, but always after a - // bound check: - uint8_t inven_icon, buttonpalette; - bool over_shoulder_on; - uint8_t show_empty_weapon; - - bool jetpack_on, spritebridge; - uint8_t lastrandomspot; // unused - bool scuba_on; - uint8_t footprintpal; - bool heat_on; - uint8_t invdisptime; - - bool holster_weapon; - uint8_t falling_counter, footprintshade; - uint8_t refresh_inventory; - const uint8_t last_full_weapon; - - const uint8_t toggle_key_flag; - uint8_t knuckle_incs, knee_incs, access_incs; - uint8_t walking_snd_toggle, palookup, hard_landing, fist_incs; - - int8_t numloogs, loogcnt; - const int8_t scream_voice; - const int8_t last_weapon; - const int8_t cheat_phase; - int8_t weapon_pos; - const int8_t wantweaponfire; - const int8_t curr_weapon; - - const uint8_t palette; - palette_t _pals; - int8_t _palsfadespeed, _palsfadenext, _palsfadeprio, _padding2; - - // NOTE: In C, the struct type has no name. We only have it here to define - // a metatype later. - const weaponaccess_t weapon; - const int8_t _padding; -} -]] - -local PROJECTILE_STRUCT = [[ -struct { - int32_t workslike, cstat; - int32_t hitradius, range, flashcolor; - const int16_t spawns; - const int16_t sound, isound; - int16_t vel; - const int16_t decal, trail; - int16_t tnum, drop; - int16_t offset, bounces; - const int16_t bsound; - int16_t toffset; - int16_t extra, extra_rand; - int8_t sxrepeat, syrepeat, txrepeat, tyrepeat; - int8_t shade, xrepeat, yrepeat, pal; - int8_t movecnt; - uint8_t clipdist; - int8_t filler[2]; - int32_t userdata; -} -]] - --- KEEPINSYNC weapondata_mt below. -local WEAPONDATA_STRUCT = "struct {"..table.concat(con_lang.wdata_members, ';').."; }" - -local randgen = require("randgen") - -local ma_rand = randgen.new(true) -- initialize to "random" (time-based) seed -local ma_count = nil - -local function ma_replace_array(typestr, neltstr) - local nelts = tonumber(neltstr) - if (nelts==nil) then - nelts = ffiC[neltstr] - assert(type(nelts)=="number") - end - - local strtab = { "const ", typestr.." " } - for i=1,nelts do - local ch1 = 97 + (ma_rand:getu32() % 25) -- 'a'..'z' - strtab[i+2] = string.format("_%c%x%s", ch1, ma_count, (i] = true if setting <0 allowed -local DukePlayer_prot_chkfunc = {} -- [] = - -local function ma_replace_scalar(what, typestr, membname) - DukePlayer_prot_chkfunc[membname] = assert(prot_scalar_chkfunc[what:sub(1,1)]) - DukePlayer_prot_allowneg[membname] = (what:sub(2)=="-") - return ma_replace_array(typestr, 1) -end - --- Converts a template struct definition to an external one, in which arrays --- have been substituted by randomly named scalar fields. --- : also handle protected scalars like "const ..." etc. -local function mangle_arrays(structstr, also_scalars) - ma_count = 0 - -- NOTE: regexp only works for non-nested arrays and for one array per line. - structstr = structstr:gsub("const%s+([%w_]+)[^\n]+%[([%w_]+)%];", ma_replace_array) - - if (also_scalars) then - -- One protected scalar per line, too. - structstr = structstr:gsub("const<(.-)>%s+([%w_]-)%s+([%w_]-);", ma_replace_scalar) - end - - return structstr -end - ---print(mangle_arrays(DUKEPLAYER_STRUCT, true)) - ---- default defines etc. -local con_lang = require("con_lang") -local xmath = require("xmath") - -ffi.cdef([[ -typedef struct { int32_t _p; } weaponaccess_t; - -typedef struct { -]]..defs_c.bitint_member("UBit32", "bits")..[[ - int16_t fvel, svel, avel; - int8_t horz, extbits; -} input_t; - -typedef -]].. mangle_arrays(ACTOR_STRUCT) ..[[ -actor_t; - -typedef -]].. mangle_arrays(DUKEPLAYER_STRUCT, true) ..[[ -DukePlayer_t; - -typedef __attribute__((packed)) struct { - DukePlayer_t *ps; - input_t *sync; - - int32_t netsynctime; - int16_t ping, filler; - int32_t pcolor, pteam; - uint8_t frags[MAXPLAYERS], wchoice[MAX_WEAPONS]; - - char vote, gotvote, pingcnt, playerquitflag, ready; - char user_name[32]; - uint32_t revision; -} playerdata_t; - -typedef struct { - int32_t cur, count; - int32_t gunposx, lookhalfang; - int32_t gunposy, lookhoriz; - int32_t shade; -} hudweapon_t; - -typedef -]].. WEAPONDATA_STRUCT ..[[ -weapondata_t; - -typedef -]].. PROJECTILE_STRUCT ..[[ -projectile_t; - -typedef struct { - uint32_t _flags; // XXX: do we want to have this accessible at game time? - int32_t _cacherange; - projectile_t *_proj; - const projectile_t *_defproj; -} tiledata_t; - -typedef struct { - vec3_t pos; - int32_t dist, clock; - int16_t ang, horiz; - int16_t sect; // NOTE: protected in camera_mt's __newindex -} camera_t; - -enum -{ - MAXMOUSEBUTTONS = 10, - MAXMOUSEAXES = 2, - MAXJOYBUTTONS = 32, - MAXJOYBUTTONSANDHATS = (32+4), - MAXJOYAXES = 9, - - NUMGAMEFUNCTIONS = 56, - - // game.h - MAXRIDECULE = 10, - MAXRIDECULELENGTH = 40, - MAXSAVEGAMENAME = 22, - MAXPWLOCKOUT = 128, - MAXRTSNAME = 128, -}; - -typedef struct { - int32_t const_visibility,uw_framerate; - int32_t camera_time,folfvel,folavel,folx,foly,fola; - int32_t reccnt,crosshairscale; - - int32_t runkey_mode,statusbarscale,mouseaiming,weaponswitch,drawweapon; // JBF 20031125 - int32_t democams,color,msgdisptime,statusbarmode; - int32_t m_noexits,noexits,autovote,automsg,idplayers; - int32_t team, viewbob, weaponsway, althud, weaponscale, textscale; - - int32_t entered_name,screen_tilting,shadows,fta_on,executions,auto_run; - int32_t coords,tickrate,levelstats,m_coop,coop,screen_size,lockout,crosshair; - int32_t playerai,angleinterpolation,obituaries; - - int32_t respawn_monsters,respawn_items,respawn_inventory,recstat,monsters_off,brightness; - int32_t m_respawn_items,m_respawn_monsters,m_respawn_inventory,m_recstat,m_monsters_off,detail; - int32_t m_ffire,ffire,m_player_skill,m_level_number,m_volume_number,multimode; - int32_t player_skill,level_number,volume_number,m_marker,marker,mouseflip; - - vec2_t m_origin; - int32_t playerbest; - - int32_t configversion, bgstretch; - - int16_t pause_on,from_bonus; - int16_t camerasprite,last_camsprite; - int16_t last_level,secretlevel; - - struct { - int32_t UseJoystick; - int32_t UseMouse; - int32_t AutoAim; - int32_t ShowOpponentWeapons; - int32_t MouseDeadZone,MouseBias; - int32_t SmoothInput; - - // JBF 20031211: Store the input settings because - // (currently) mact can't regurgitate them - int32_t MouseFunctions[MAXMOUSEBUTTONS][2]; - int32_t MouseDigitalFunctions[MAXMOUSEAXES][2]; - int32_t MouseAnalogueAxes[MAXMOUSEAXES]; - int32_t MouseAnalogueScale[MAXMOUSEAXES]; - int32_t JoystickFunctions[MAXJOYBUTTONSANDHATS][2]; - int32_t JoystickDigitalFunctions[MAXJOYAXES][2]; - int32_t JoystickAnalogueAxes[MAXJOYAXES]; - int32_t JoystickAnalogueScale[MAXJOYAXES]; - int32_t JoystickAnalogueDead[MAXJOYAXES]; - int32_t JoystickAnalogueSaturate[MAXJOYAXES]; - uint8_t KeyboardKeys[NUMGAMEFUNCTIONS][2]; - - // - // Sound variables - // - int32_t MasterVolume; - int32_t FXVolume; - int32_t MusicVolume; - int32_t SoundToggle; - int32_t MusicToggle; - int32_t VoiceToggle; - int32_t AmbienceToggle; - - int32_t NumVoices; - int32_t NumChannels; - int32_t NumBits; - int32_t MixRate; - - int32_t ReverseStereo; - - // - // Screen variables - // - - int32_t ScreenMode; - - int32_t ScreenWidth; - int32_t ScreenHeight; - int32_t ScreenBPP; - - int32_t ForceSetup; - int32_t NoAutoLoad; - - const int32_t scripthandle; - int32_t setupread; - - int32_t CheckForUpdates; - int32_t LastUpdateCheck; - int32_t useprecache; - } config; - - char overhead_on,last_overhead,showweapons; - char god,warp_on,cashman,eog,showallmap; - char show_help,scrollmode,noclip; - char ridecule[MAXRIDECULE][MAXRIDECULELENGTH]; - char pwlockout[MAXPWLOCKOUT],rtsname[MAXRTSNAME]; - char display_bonus_screen; - char show_level_text; - char wchoice[MAX_WEAPONS]; -} user_defs; - -typedef struct { - int32_t partime, designertime; - char *name, *filename, *musicfn; - void *savedstate; -} map_t; -]]) - -bcarray.new("weapondata_t", ffiC.MAX_WEAPONS, "weapon", "weapondata_x_MAX_WEAPONS", WEAPON_NAMES) -bcarray.new("int32_t", con_lang.MAXSESSIONVARS, "sessionvar", "int32_x_MAXSESSIONVARS") - --- EXTERNALLY EXPOSED GAME VARIABLES -ffi.cdef[[ -const int32_t screenpeek; -hudweapon_t hudweap; -int32_t g_logoFlags; -]] - --- INTERNAL VARIABLES/FUNCTIONS -decl("map_t g_mapInfo[$*$];", con_lang.MAXVOLUMES+1, con_lang.MAXLEVELS) -decl("char g_volumeNames[$][33];", con_lang.MAXVOLUMES) - -decl[[ -const int32_t myconnectindex; -int32_t g_RETURN; -int32_t g_elCONSize; -char *g_elCON; -void El_SetCON(const char *conluacode); -void El_OnError(const char *str); - -char *g_elSavecode; -void El_FreeSaveCode(void); -const char *(*El_SerializeGamevars)(int32_t *slenptr, int32_t levelnum); -int32_t (*El_RestoreGamevars)(const char *savecode); -int32_t (*El_GetLabelValue)(const char *label); - -const char *s_buildRev; -const char *g_sizes_of_what[]; -int32_t g_sizes_of[]; -int32_t g_elFirstTime; -int32_t g_elCallDepth; -int32_t block_deletesprite; -const char **g_elModules; -char g_modDir[]; -int32_x_MAXSESSIONVARS g_elSessionVar; -actor_t actor[MAXSPRITES]; -camera_t g_camera; -user_defs ud; -playerdata_t *const g_player; -DukePlayer_t *g_player_ps[MAXPLAYERS]; -weapondata_x_MAX_WEAPONS g_playerWeapon[MAXPLAYERS]; -weapondata_t g_weaponOverridden[MAX_WEAPONS]; -int16_t WeaponPickupSprites[MAX_WEAPONS]; -tiledata_t g_tile[MAXTILES]; -projectile_t SpriteProjectile[MAXSPRITES]; - -int32_t g_noResetVars; -void (*A_ResetVars)(int32_t iActor); - -// Used from lunacon.lua for dynamic {tile,sound} remapping: -struct -{ - const char *str; - int32_t *dynvalptr; - const int16_t staticval; -} g_dynTileList[], g_dynSoundList[]; - -char *apStrings[]; - -const int32_t g_mostConcurrentPlayers; -int16_t g_deleteQueueSize; -int16_t g_blimpSpawnItems[15]; -int32_t g_scriptVersion; -const int32_t g_frameRate; -const int32_t g_currentMenu; -uint16_t g_earthquakeTime; -uint32_t g_moveThingsCount; -char CheatKeys[2]; - -// Must not have functions here that may call events directly or -// indirectly. See lunatic_game.c. - -int32_t A_IncurDamage(int32_t sn); // not bound-checked! -int32_t G_CheckActivatorMotion(int32_t lotag); -int32_t A_Dodge(spritetype *s); -int32_t A_MoveSpriteClipdist(int32_t spritenum, const vec3_t *change, uint32_t cliptype, int32_t clipdist); -void P_DoQuote(int32_t q, DukePlayer_t *p); -void P_SetGamePalette(DukePlayer_t *player, uint32_t palid, int32_t set); -void G_AddUserQuote(const char *daquote); -void G_ClearCameraView(DukePlayer_t *ps); -void VM_DrawTileGeneric(int32_t x, int32_t y, int32_t zoom, int32_t tilenum, - int32_t shade, int32_t orientation, int32_t p); -void G_InitTimer(int32_t ticspersec); -void G_GetTimeDate(int32_t *vals); -int32_t G_ToggleWallInterpolation(int32_t w, int32_t doset); -int32_t G_StartTrack(int32_t level); -int32_t VM_CheckSquished2(int32_t i, int32_t snum); -void Menu_Change(int32_t cm); - -const char *KB_ScanCodeToString(uint8_t scancode); - -int32_t A_CheckAnySoundPlaying(int32_t i); -int32_t S_CheckSoundPlaying(int32_t i, int32_t num); -void S_StopEnvSound(int32_t num, int32_t i); -void S_StopAllSounds(void); -void S_ChangeSoundPitch(int32_t num, int32_t i, int32_t pitchoffset); -int32_t S_GetMusicPosition(void); -void S_SetMusicPosition(int32_t position); - -int32_t minitext_(int32_t x,int32_t y,const char *t,int32_t s,int32_t p,int32_t sb); -void G_DrawTXDigiNumZ(int32_t starttile, int32_t x,int32_t y,int32_t n,int32_t s,int32_t pal, - int32_t cs,int32_t x1, int32_t y1, int32_t x2, int32_t y2, int32_t z); -void G_PrintGameText(int32_t tile, int32_t x, int32_t y, const char *t, - int32_t s, int32_t p, int32_t o, - int32_t x1, int32_t y1, int32_t x2, int32_t y2, - int32_t z, int32_t a); -vec2_t G_ScreenText(const int32_t font, - int32_t x, int32_t y, const int32_t z, const int32_t blockangle, const int32_t charangle, - const char *str, const int32_t shade, int32_t pal, int32_t o, int32_t alpha, - int32_t xspace, int32_t yline, int32_t xbetween, int32_t ybetween, const int32_t f, - const int32_t x1, const int32_t y1, const int32_t x2, const int32_t y2); -vec2_t G_ScreenTextSize(const int32_t font, - int32_t x, int32_t y, const int32_t z, const int32_t blockangle, - const char *str, const int32_t o, - int32_t xspace, int32_t yline, int32_t xbetween, int32_t ybetween, - const int32_t f, - int32_t x1, int32_t y1, int32_t x2, int32_t y2); -const char* G_PrintYourTime(void); -const char* G_PrintParTime(void); -const char* G_PrintDesignerTime(void); -const char* G_PrintBestTime(void); - -void G_UpdateScreenArea(void); -void G_SaveMapState(void); -void G_RestoreMapState(void); -void G_FreeMapState(int32_t mapnum); -]] - -decl[[ -int32_t kopen4loadfrommod(const char *filename, char searchfirst); - -char **g_scriptModules; -int32_t g_scriptModulesNum; - -const char *G_ConFile(void); -void G_DoGameStartup(const int32_t *params); -int32_t C_DefineSound(int32_t sndidx, const char *fn, int32_t args [5]); -void C_DefineMusic(int32_t vol, int32_t lev, const char *fn); -void C_DefineQuote(int32_t qnum, const char *qstr); -void C_DefineVolumeName(int32_t vol, const char *name); -void C_DefineVolumeFlags(int32_t vol, int32_t flags); -void C_UndefineVolume(int32_t vol); -void C_DefineSkillName(int32_t skill, const char *name); -void C_UndefineSkill(int32_t skill); -void C_DefineLevelName(int32_t vol, int32_t lev, const char *fn, - int32_t partime, int32_t designertime, - const char *levelname); -void C_UndefineLevel(int32_t vol, int32_t lev); -void C_DefineProjectile(int32_t j, int32_t what, int32_t val); -void C_DefineGameFuncName(int32_t idx, const char *name); -void C_DefineGameType(int32_t idx, int32_t flags, const char *name); -int32_t C_SetDefName(const char *name); -void C_SetCfgName(const char *cfgname); - -int32_t SCRIPT_GetNumber(int32_t scripthandle, const char *sectionname, const char *entryname, int32_t *number); -void SCRIPT_PutNumber(int32_t scripthandle, const char *sectionname, const char *entryname, int32_t number, - int32_t hexadecimal, int32_t defaultvalue); -]] - - --- http://lua-users.org/wiki/SandBoxes says "potentially unsafe" --- as it allows to see implementations of functions. ---local string_dump = string.dump -string.dump = nil - - --- sanity-check struct type sizes -local good = true -for i=0,10 do - local what = ffi.string(ffiC.g_sizes_of_what[i]) - local fsz = ffi.sizeof(what) - local csz = ffiC.g_sizes_of[i] - if (ffiC._DEBUG_LUNATIC ~= 0) then - print(i..": "..what..": C sizeof = "..tostring(csz)..", FFI sizeof = "..tostring(fsz)) - end - if (fsz ~= csz) then - good = false; - end -end - -if (not good) then - error("Some sizes don't match between C and LuaJIT/FFI.") -end - - ---== "player" global, needed by the "control" module ==-- - -local player_static_members = defs_c.static_members_tab() - ---[[ -player_static_members._INPUT_BITS = defs_c.conststruct -{ - JUMP = 1, - CROUCH = 2, - FIRE = 4, - AIM_UP = 8, - AIM_DOWN = 16, - RUNNING = 32, - LOOK_LEFT = 64, - LOOK_RIGHT = 128, - -- weapons... - STEROIDS = 4096, - LOOK_UP = 8192, - LOOK_DOWN = 16384, - NIGHTVISION = 32768, - MEDKIT = 65536, - RESERVED = 131072, - CENTER_VIEW = 262144, - HOLSTER_WEAPON = 524288, - INVENTORY_LEFT = 1048576, - PAUSE = 2097152, - QUICK_KICK = 4194304, - AIM_MODE = 8388608, - HOLODUKE = 16777216, - JETPACK = 33554432, - QUIT = 67108864, - INVENTORY_RIGHT = 134217728, - TURN_AROUND = 268435456, - OPEN = 536870912, - INVENTORY = 1073741824, - ESC = 2147483648, -} - -player_static_members._INPUT_EXT_BITS = defs_c.conststruct -{ - MOVE_FORWARD = 1, - MOVE_BACKWARD = 2, - STRAFE_LEFT = 4, - STRAFE_RIGHT = 8, - TURN_LEFT = 16, - TURN_RIGHT = 32, -} ---]] - -local band = bit.band -local lsh = bit.lshift -local ivec3 = xmath.ivec3 - -do - -- player.all() iterator - local function iter_allplayers(_nil, pli) - if (pli+1 < ffiC.g_mostConcurrentPlayers) then - return pli+1 - end - end - - function player_static_members.all() - return iter_allplayers, nil, -1 - end - - -- player.holdskey(pli, keyname) - local KEYS = { -- SK_CROUCH etc. -- "sync keys" - CROUCH = lsh(1,1), - RUN = lsh(1,5), - OPEN = lsh(1,29), - } - - function player_static_members.holdskey(pli, keyname) - bcheck.player_idx(pli) - if (KEYS[keyname] == nil) then - error("invalid key name: "..tostring(keyname), 2) - end - - return ffiC.g_player[pli].sync.bitsbits:test(KEYS[keyname]) - end -end - -local player_holdskey = player_static_members.holdskey - --- Actor flags -local actor_static_members = defs_c.static_members_tab() -do - local our_SFLAG = {} - local ext_SFLAG = con_lang.labels[4] -- external actor flags only - local USER_MASK = 0 - - for name, flag in pairs(ext_SFLAG) do - our_SFLAG[name:sub(7)] = flag -- strip "SFLAG_" - USER_MASK = bit.bor(USER_MASK, flag) - end - - -- Add a couple of convenience defines. - our_SFLAG.enemy = con_lang.SFLAG.SFLAG_BADGUY - our_SFLAG.enemystayput = con_lang.SFLAG.SFLAG_BADGUY + con_lang.SFLAG.SFLAG_BADGUYSTAYPUT - our_SFLAG.rotfixed = con_lang.SFLAG.SFLAG_ROTFIXED - - -- Callback function chaining control flags. - our_SFLAG.replace = 0x08000000 - our_SFLAG.replace_soft = 0x08000000 -- compat - our_SFLAG.replace_hard = 0x08000000 -- compat, deprecated - our_SFLAG.chain_beg = 0x20000000 - our_SFLAG.chain_end = 0x40000000 - our_SFLAG._CHAIN_MASK_ACTOR = 0x78000000 - our_SFLAG._CHAIN_MASK_EVENT = 0x68000000 - - -- XXX: CON doesn't export BADGUYSTAYPUT or ROTFIXED SFLAGs, but they are considered - -- external for Lunatic. - our_SFLAG.USER_MASK = bit.bor(USER_MASK, our_SFLAG.enemystayput, our_SFLAG.rotfixed) - - actor_static_members.FLAGS = defs_c.conststruct(our_SFLAG) - - -- Sprite status numbers. Kept in 'actor', because it's more of a game-side - -- concept (event though status lists are implemented in the engine), and - -- to prevent confusion with sprite.CSTAT. - local our_STAT = {} - - for name, statnum in pairs(con_lang.STAT) do - -- Strip 'STAT_'. - our_STAT[name:sub(6)] = statnum - end - - actor_static_members.STAT = defs_c.conststruct(our_STAT) - - actor_static_members.MOVFLAGS = defs_c.conststruct - { - -- NOTE: no underscores, like in DEFS.CON. - faceplayer = 1, - geth = 2, - getv = 4, - randomangle = 8, - faceplayerslow = 16, - spin = 32, - faceplayersmart = 64, - fleeenemy = 128, - jumptoplayer_only = 256, - jumptoplayer_bits = 257, -- NOTE: two bits set! - jumptoplayer = 257, - seekplayer = 512, - furthestdir = 1024, - dodgebullet = 4096, - } -end - -function actor_static_members.fall(i) - check_sprite_idx(i) - CF.VM_FallSprite(i) -end - --- actor.move(i, vec, cliptype [, clipdist]) -function actor_static_members.move(i, vec, cliptype, clipdist) - check_sprite_idx(i) - local vel = ivec3(vec.x, vec.y, vec.z) - return ffiC.A_MoveSpriteClipdist(i, vel, cliptype, clipdist or -1) -end - --- Delete sprite with index . -function actor_static_members.delete(i) - check_sprite_idx(i) - - if (ffiC.sprite[i].statnum == ffiC.MAXSTATUS) then - error("Attempt to delete a sprite already not in the game world", 2) - end - - if (ffiC.block_deletesprite ~= 0) then - error("Attempt to delete sprite in EVENT_EGS", 2) - end - - -- TODO_MP - if (ffiC.g_player_ps[0].i == i) then - error("Attempt to delete player 0's APLAYER sprite", 2) - end - - CF.A_DeleteSprite(i) -end - -local tile_static_members = defs_c.static_members_tab() -do - tile_static_members.sizx = defs_c.creategtab_membidx(ffiC.tilesiz, "x", ffiC.MAXTILES, "tilesizx[]") - tile_static_members.sizy = defs_c.creategtab_membidx(ffiC.tilesiz, "y", ffiC.MAXTILES, "tilesizy[]") -end - --- XXX: error message will say "g_player_ps" -player = setmtonce({}, defs_c.GenStructMetatable("g_player_ps", "g_mostConcurrentPlayers", player_static_members)) - --- needed by "control" -actor = setmtonce({}, defs_c.GenStructMetatable("actor", "MAXSPRITES", actor_static_members)) --- Some bitwise NOTs of various actor flag masks. -local BNOT = { - USER_MASK = bit.bnot(actor.FLAGS.USER_MASK), - CHAIN_MASK_ACTOR = bit.bnot(actor.FLAGS._CHAIN_MASK_ACTOR), - CHAIN_MASK_EVENT = bit.bnot(actor.FLAGS._CHAIN_MASK_EVENT), -} - -local projectile = defs_c.creategtab_membidx_ptr(ffiC.g_tile, "_proj", ffiC.MAXTILES, "projectile") -local g_tile = setmtonce({}, defs_c.GenStructMetatable("g_tile", "MAXTILES", tile_static_members)) - ---== Custom operations for BUILD data structures ==-- --- Among other things, declares struct action and struct move, and their --- ID-wrapped types con_action_t and con_move_t -local con = require("control") - -do - local isenemytile = con.isenemytile - - -- Add game-side metamethods to "spritetype" and register it with "metatype" - local spr_mt_index_add = { - isenemy = function(s) - return isenemytile(s.picnum) - end, - } - - defs_c.finish_spritetype(spr_mt_index_add) -end - --- Check a literal numeric action or move value. -local function check_literal_am(am, typename) - if (type(am) ~= "number") then - error("bad argument: expected number or "..typename, 3) - end - - -- Negative values are generated as con.action/con.move IDs. - if (not (am >= 0 and am <= 32767)) then - error("bad argument: expected number in [0 .. 32767]", 3) - end -end - --- An unrestricted actor_t pointer, for internal use: -local actor_ptr_ct = ffi.typeof("$ *", ffi.typeof(strip_const(ACTOR_STRUCT))) -local player_ptr_ct = ffi.typeof("$ *", ffi.typeof(strip_const(DUKEPLAYER_STRUCT))) -local projectile_ptr_ct = ffi.typeof("$ *", ffi.typeof(strip_const(PROJECTILE_STRUCT))) --- An unrestricted weapondata_t pointer, but with the member names stripped of --- the leading underscore, too: -local weapondata_ptr_ct = ffi.typeof("$ *", ffi.typeof((strip_const(WEAPONDATA_STRUCT):gsub(" _"," ")))) - -local con_action_ct = ffi.typeof("const con_action_t") -local con_move_ct = ffi.typeof("const con_move_t") -local con_ai_ct = ffi.typeof("const con_ai_t") - --- All-zero bare action and move. -local nullac, nullmv = ffi.new("const struct action"), ffi.new("const struct move") --- All-zero action and move with IDs. Mostly for CON support. -local literal_act = { [0]=con_action_ct(0), [1]=con_action_ct(1) } -local literal_mov = { [0]=con_move_ct(0), [1]=con_move_ct(1) } - -local function get_actor_idx(a) - local i = ffi.cast(actor_ptr_ct, a)-ffi.cast(actor_ptr_ct, ffiC.actor) --- assert(not (i >= ffiC.MAXSPRITES+0ULL)) - return i -end - -local actor_methods = { - -- action - set_action = function(a, act) - a = ffi.cast(actor_ptr_ct, a) - - if (ffi.istype(con_action_ct, act)) then - a.t_data[4] = act.id - a.ac = act.ac - else - check_literal_am(act, "action") - a.t_data[4] = act - a.ac = nullac - end - - a.t_data[2] = 0 - a.t_data[3] = 0 - end, - - has_action = function(a, act) - a = ffi.cast(actor_ptr_ct, a) - - if (ffi.istype(con_action_ct, act)) then - return (a.t_data[4]==act.id) - else - check_literal_am(act, "action") - return (a.t_data[4]==act) - end - end, - - -- count - set_count = function(a, count) - ffi.cast(actor_ptr_ct, a).t_data[0] = count - end, - - get_count = function(a) - return ffi.cast(actor_ptr_ct, a).t_data[0] - end, - - -- action count - reset_acount = function(a) - ffi.cast(actor_ptr_ct, a).t_data[2] = 0 - end, - - get_acount = function(a) - return ffi.cast(actor_ptr_ct, a).t_data[2] - end, - - -- Override action delay. The action ID is kept. - set_action_delay = function(a, delay) - ffi.cast(actor_ptr_ct, a).ac.delay = delay - end, - - -- move - set_move = function(a, mov, movflags) - a = ffi.cast(actor_ptr_ct, a) - - if (ffi.istype(con_move_ct, mov)) then - a.t_data[1] = mov.id - a.mv = mov.mv - else - check_literal_am(mov, "move") - a.t_data[1] = mov - a.mv = nullmv - end - - a.t_data[0] = 0 - a.movflags = movflags or 0 - local spr = ffiC.sprite[get_actor_idx(a)] - - if (not spr:isenemy() or spr.extra > 0) then - if (bit.band(a.movflags, 8) ~= 0) then -- random_angle - spr.ang = bit.band(ffiC.krand(), 2047) - end - end - end, - - has_move = function(a, mov) - a = ffi.cast(actor_ptr_ct, a) - - if (ffi.istype(con_move_ct, mov)) then - return (a.t_data[1]==mov.id) - else - check_literal_am(mov, "move") - return (a.t_data[1]==mov) - end - end, - - -- Override velocity, keeping move ID. - set_hvel = function(a, hvel) - ffi.cast(actor_ptr_ct, a).mv.hvel = hvel - end, - - set_vvel = function(a, vvel) - ffi.cast(actor_ptr_ct, a).mv.vvel = vvel - end, - - -- ai - set_ai = function(a, ai) - local oa = a - a = ffi.cast(actor_ptr_ct, a) - - -- TODO: literal number AIs? - if (not ffi.istype(con_ai_ct, ai)) then - error("bad argument: expected ai", 2) - end - - -- NOTE: compare with gameexec.c, "CON_AI:" - a.t_data[5] = ai.id - - oa:set_action(ai.act) - oa:set_move(ai.mov, ai.movflags) - - -- Already reset with set_move(): --- a.t_data[0] = 0 - end, - - has_ai = function(a, ai) - a = ffi.cast(actor_ptr_ct, a) - - if (ffi.istype(con_ai_ct, ai)) then - return (a.t_data[5]==ai.id) - else - check_literal_am(ai, "ai") - return (a.t_data[5]==ai) - end - end, - - -- Getters/setters. - _get_t_data = function(a, idx) - if (not (idx >= 0 and idx < 10)) then - error("invalid t_data index "..idx, 2) - end - return ffi.cast(actor_ptr_ct, a).t_data[idx] - end, - - _set_t_data = function(a, idx, val) - if (not (idx >= 0 and idx < 10)) then - error("invalid t_data index "..idx, 2) - end - ffi.cast(actor_ptr_ct, a).t_data[idx] = val - end, - - set_picnum = function(a, picnum) - if (not (picnum < 0)) then - check_tile_idx(picnum) - end - ffi.cast(actor_ptr_ct, a).picnum = picnum - end, - - set_owner = function(a, owner) - -- XXX: is it permissible to set to -1? - check_sprite_idx(owner) - ffi.cast(actor_ptr_ct, a).owner = owner - end, - - --- Custom methods --- - - -- Checkers for whether the movement update made the actor hit - -- something. - - checkhit = function(a) - -- Check whether we hit *anything*, including ceiling/floor. - return a._movflagbits:test(49152) - end, - - checkbump = function(a) - -- Check whether we bumped into a wall or sprite. - return (a._movflagbits:mask(49152) >= 32768) - end, - - hitwall = function(a) - if (a._movflagbits:mask(49152) == 32768) then - return a._movflagbits:mask(32767) - end - end, - - hitsprite = function(a) - if (a._movflagbits:mask(49152) == 49152) then - return a._movflagbits:mask(32767) - end - end, - - -- NOTE: no 'hitsector' or 'hitceiling' / 'hitfloor' for now because - -- more research is needed as to what the best way of checking c/f is. -} - -local actor_mt = { - __index = function(a, key) - if (actor_methods[key] ~= nil) then - return actor_methods[key] - elseif (key == "proj") then - return ffiC.SpriteProjectile[get_actor_idx(a)] - else - error("invalid indexing key to actor object", 2) - end - end, -} - -ffi.metatype("actor_t", actor_mt) - - ---- PER-PLAYER WEAPON SETTINGS -local wd_sound_member = {} -for _, declstr in pairs(con_lang.wdata_members) do - local member = declstr:match("const int32_t _(.*sound)$") - if (member) then - wd_sound_member[member] = true - if (ffiC._DEBUG_LUNATIC ~= 0) then - printf("weapondata_t member %s denotes a sound", member) - end - end -end - -local weapondata_mt = { - __index = function(wd, member) - -- Handle protected members that are renamed (e.g. shoots/_shoots). - return ffi.cast(weapondata_ptr_ct, wd)[0][member] - end, - - __newindex = function(wd, member, val) - -- Set to 'true' if we set a tile or sound member. - local didit = false - - check_type(member, "string") -- MEMBER_IS_STRING - check_number(val) - - if (wd_sound_member[member]) then -- XXX: sound2time is a time, not a sound - if (val < 0) then - val = 0 -- Set to 0 if negative (e.g. CrackDown). - else - check_sound_idx(val) - end - didit = true - elseif (member=="workslike") then - check_weapon_idx(val) - elseif (member=="spawn" or member=="shoots") then - if (val < 0) then - -- Set to 0 if oob (e.g. CrackDown). This is a bit problematic - -- for .shoots because it's used unconditionally except in one - -- case (see player.c). - val = 0 - else - check_tile_idx(val) - end - didit = true - end - - -- DEBUG: --- printf("assigning %s to weapon's %s", tostring(val), member) - - -- NOTE: we're indexing a *pointer* with the user-supplied 'member', - -- which could be dangerouns if it could be a number. However, we have - -- assured that is is not in MEMBER_IS_STRING above. - ffi.cast(weapondata_ptr_ct, wd)[member] = val - - if (didit and ffiC.g_elCallDepth==0) then - -- Signal that we overrode this member at CON translation time. - - -- Get weapon index as pointer difference first. PLAYER_0. - local wi = ffi.cast(weapondata_ptr_ct, wd) - - ffi.cast(weapondata_ptr_ct, ffiC.g_playerWeapon) - assert(wi >= 0 and wi < ffiC.MAX_WEAPONS) - - -- Set g_weaponOverridden[wi][member], but without invoking - -- weapondata_t's __newindex metamethod (i.e. us)! - ffi.cast(weapondata_ptr_ct, ffiC.g_weaponOverridden[wi])[member] = 1 - end - end, -} -ffi.metatype("weapondata_t", weapondata_mt) - -local weaponaccess_mt = { - -- Syntax like "player[0].weapon.KNEE.shoots" possible because - -- g_playerWeapon[] is declared as an array of corresponding bcarray types - -- for us. - __index = function(wa, key) - if (type(key)~="number" and type(key)~="string") then - error("must access weapon either by number or by name") - end - - return ffiC.g_playerWeapon[wa._p][key] - end, -} -ffi.metatype("weaponaccess_t", weaponaccess_mt) - - -local function clamp(num, min, max) - return num < min and min - or num > max and max - or num -end - -local function clamp0to1(num) - check_number(num, 4) - return clamp(num, 0, 1) -end - -local player_methods = { - -- CON-like addammo/addweapon, but without the non-local control flow - -- (returns true if weapon's ammo was at the max. instead). - addammo = con._addammo, - addweapon = con._addweapon, - - stomp = con._pstomp, - - holdskey = function(p, keyname) - -- XXX: on invalid , error will point to this next line: - return player_holdskey(p.weapon._p, keyname) - end, - - has_weapon = function(p, weap) - return p.gotweaponbits:test(lsh(1,weap)) - end, - - give_weapon = function(p, weap) - p.gotweaponbits:set(lsh(1,weap)) - end, - - take_weapon = function(p, weap) - p.gotweaponbits:clear(lsh(1,weap)) - end, - - -- Give or take weapon, for implementation of CON's .gotweapon access. - _gt_weapon = function(p, weap, give_p) - if (give_p ~= 0) then - p:give_weapon(weap) - else - p:take_weapon(weap) - end - end, - - whack = function(p, no_return_to_center) - p.horiz = p.horiz + 64 - if (not no_return_to_center) then - p.return_to_center = 9 - end - local n = bit.arshift(128-band(ffiC.krand(),255), 1) - p.rotscrnang = n - p.look_ang = n - end, - - -- External, improved 'palfrom'. - -- : possibly fractional speed of tint fading, in pals.f decrements per gametic. - -- XXX: exposes internals. - -- : a value from -128 to 127, higher ones trump lower or equal ones - fadecol = function(p, fadefrac, r, g, b, speed, prio) - -- Validate inargs: clamp f,r,g,b to [0 .. 1] first and multiply by - -- 63 for the internal handling. - fadefrac = clamp0to1(fadefrac)*63 - -- NOTE: a fadefrac of now <1 is allowed, e.g. for clearing the tint. - r = clamp0to1(r)*63 - g = clamp0to1(g)*63 - b = clamp0to1(b)*63 - - if (speed ~= nil) then - check_number(speed) - -- Clamp to sensible values; the speed is stored in an int8_t - -- (see below). - speed = clamp(speed, 1/128, 127) - else - speed = 1 - end - - if (prio ~= nil) then - check_number(prio) - - if (not (prio >= -128 and prio < 127)) then - error("invalid argument #6 (priority): must be in [-128 .. 127]", 2) - end - else - prio = 0 - end - - -- Check if a currently active tint has higher priority. - if (p._pals.f > 0 and p._palsfadeprio > prio) then - return - end - - -- The passed tint can be applied now. - p:_palfrom(fadefrac, r, g, b) - p._palsfadeprio = prio - - -- Calculate .palsfade{speed,next} - if (speed >= 1) then - -- Will round to the nearest integer ("banker's - -- rounding"). NOTE: This is not correct for all numbers, see - -- http://blog.frama-c.com/index.php?post/2013/05/02/nearbyintf1 - p._palsfadespeed = speed + 0.5 - p._palsfadenext = 0 - else - -- NOTE: Values that round to 0 have are equivalent behavior to - -- passing a of 1. - local negspeedrecip = -((1/speed) + 0.5) -- [-128.5 .. 1/127+0.5] - p._palsfadespeed = negspeedrecip - p._palsfadenext = negspeedrecip - end - end, - - -- INTERNAL and CON-only. - _palfrom = function(p, f, r,g,b) - local pals = p._pals - -- Assume that CON's palfrom starts with prio 0 and speed 0. - if (pals.f == 0 or p._palsfadeprio <= 0) then - pals.f = f - pals.r, pals.g, pals.b = r, g, b - p._palsfadespeed, p._palsfadenext = 0, 0 - end - end, -} - -local player_mt = { - __index = function(p, key) - if (player_methods[key] ~= nil) then - return player_methods[key] - elseif (key == "_input") then - return ffiC.g_player[p.weapon._p].sync[0] - else - -- Read access to protected player members. - return ffi.cast(player_ptr_ct, p)[0][key] - end - end, - - __newindex = function(p, key, val) - -- Write access to protected player members. - - local allowneg = DukePlayer_prot_allowneg[key] - assert(type(allowneg)=="boolean") - - if (allowneg==false or not (val == -1)) then - DukePlayer_prot_chkfunc[key](val) - end - ffi.cast(player_ptr_ct, p)[key] = val - end, -} - -ffi.metatype("DukePlayer_t", player_mt) - -local function GenProjectileSetFunc(Member, checkfunc) - return function(self, idx) - if (not (idx == -1)) then - checkfunc(idx) - end - ffi.cast(projectile_ptr_ct, self)[Member] = idx - end -end - -local projectile_mt = { - __index = { - set_spawns = GenProjectileSetFunc("spawns", check_tile_idx), - set_decal = GenProjectileSetFunc("decal", check_tile_idx), - set_trail = GenProjectileSetFunc("trail", check_tile_idx), - - set_sound = GenProjectileSetFunc("sound", check_sound_idx), - set_isound = GenProjectileSetFunc("isound", check_sound_idx), - set_bsound = GenProjectileSetFunc("bsound", check_sound_idx), - }, -} -ffi.metatype("projectile_t", projectile_mt) - -local user_defs_mt = { - __index = { - set_screen_size = function(ud, screen_size) - if (ud.screen_size ~= screen_size) then - ud.screen_size = screen_size - ffiC.G_UpdateScreenArea() - end - end, - - set_volume_number = function(ud, volume_number) - -- NOTE: allow volume_number==MAXVOLUMES. - if (not (volume_number==con_lang.MAXVOLUMES)) then - bcheck.volume_idx(volume_number) - end - ud.volume_number = volume_number - end, - - set_m_volume_number = function(ud, volume_number) - -- NOTE: allow volume_number==MAXVOLUMES. - if (not (volume_number==con_lang.MAXVOLUMES)) then - bcheck.volume_idx(volume_number) - end - ud.m_volume_number = volume_number - end, - - set_level_number = function(ud, level_number) - bcheck.level_idx(level_number) - ud.level_number = level_number - end, - }, -} -ffi.metatype("user_defs", user_defs_mt) - ---- CUSTOM "gv" VARIABLES -local camera_mt = { - -- TODO: "set position" method, which also updates the sectnum - __index = ffiC.g_camera, - __newindex = function(_, key, val) - if (key=="sect") then - check_sector_idx(val) - end - ffiC.g_camera[key] = val - end, -} - -gv_access.cam = setmtonce({}, camera_mt) -gv_access._ud = ffiC.ud - --- Support for some CON global system gamevars. RETURN handled separately. -gv_access._csv = ffi.new "struct { int32_t LOTAG, HITAG, TEXTURE; }" - -gv_access.REND = defs_c.conststruct -{ - CLASSIC = 0, - POLYMOST = 3, - POLYMER = 4, -} - -gv_access.WEAPON = lprivate.WEAPON -gv_access.GET = lprivate.GET - -function gv_access._get_yxaspect() - return ffiC.yxaspect -end - -function gv_access._get_viewingrange() - return ffiC.viewingrange -end - -function gv_access._currentFramerate() - return ffiC.g_frameRate -end - -function gv_access._currentMenu() - return ffiC.g_currentMenu -end - -function gv_access._changeMenu(cm) - ffiC.Menu_Change(cm) -end - -function gv_access._set_guniqhudid(id) - local MAXUNIQHUDID = 256 -- KEEPINSYNC build.h - if (not (id >= 0 and id < MAXUNIQHUDID)) then - error("invalid unique HUD ID "..id) - end - ffiC.guniqhudid = id -end - -function gv_access.currentEpisode() - return ffiC.ud.volume_number + 1 -end - -function gv_access.currentLevel() - return ffiC.ud.level_number + 1 -end - -function gv_access.doQuake(gametics, snd) - ffiC.g_earthquakeTime = gametics - if (snd ~= nil) then - con._globalsound(ffiC.screenpeek, snd) - end -end - --- Declare all con_lang.labels constants in the global FFI namespace. -for i=1,#con_lang.labels do - if (getmetatable(con_lang.labels[i]) ~= "noffiC") then - local strbuf = {"enum {"} - for label, val in pairs(con_lang.labels[i]) do - strbuf[#strbuf+1] = string.format("%s = %d,", label, val) - end - strbuf[#strbuf+1] = "};" - - ffi.cdef(table.concat(strbuf)) - end -end - - ----=== Set up restricted global environment ===--- - -local allowed_modules = { - coroutine=coroutine, bit=bit, table=table, math=math, string=string, - - os = { - clock = function() return gv_.gethiticks()*0.001 end, - }, - - randgen = randgen, - engine = require("engine"), - stat = require("stat"), - bitar = require("bitar"), - xmath = xmath, - fs = require("fs"), - - con = con, -} - -do - local ctype_cansave = {} - - -- Register as "serializeable" the type of cdata object . - local function reg_serializable_cv(v) - assert(type(v)=="cdata") - assert(type(v._serialize)=="function") - -- NOTE: tonumber() on a ctype cdata object gets its LuaJIT-internal - -- ID, the one that would be shown with tostring(), e.g. - -- ctype - ctype_cansave[tonumber(ffi.typeof(v))] = true - end - - function lprivate.cansave_cdata(v) - return type(v)=="cdata" and ctype_cansave[tonumber(ffi.typeof(v))] - end - - reg_serializable_cv(xmath.vec3()) - reg_serializable_cv(ivec3()) -end - --- Protect base modules. -local function basemod_newidx() - error("modifying base module table forbidden", 2) -end - -for modname, themodule in pairs(allowed_modules) do - local mt = { - __index = themodule, - __newindex = basemod_newidx, - } - - allowed_modules[modname] = setmtonce({}, mt) -end - - ----=== Module stuff ===--- - -local package_loaded = {} -- [] = false/true/table -local modname_stack = {} -- []=string -local module_gamevars = {} -- [] = { , , ... } -local module_gvlocali = {} -- [] = { , } -local module_thread = {} -- [] = - -local function getcurmodname(thisfuncname) - if (#modname_stack == 0) then - error("'"..thisfuncname.."' must be called at the top level of a require'd file", 3) - -- ... as opposed to "at runtime". - end - - return modname_stack[#modname_stack] -end - - -local function errorf(level, fmt, ...) - local errmsg = string.format(fmt, ...) - error(errmsg, level+1) -end - -local function readintostr_mod(fn) - -- TODO: g_loadFromGroupOnly? - local fd = ffiC.kopen4loadfrommod(fn, 0) - if (fd < 0) then - return nil - end - - local ret = defs_c.readintostr(fd) - ffiC.kclose(fd) - return ret -end - - -local debug = require("debug") - --- Get the number of active locals in the function that calls the function --- calling this one. -local function getnumlocals(l) - -- 200 is the max. number of locals at one level - for i=1,200 do - -- level: - -- 0 is debug.getlocal() itself. - -- 1 is this function (getnumlocals). - -- 2 is the function calling getnumlocals() - -- 3 is the function calling that one. - if (debug.getlocal(3, i) == nil) then - return i-1 - end - end -end - -local function error_on_nil_read(_, varname) - error("attempt to read nil variable '"..varname.."'", 2) -end - -local required_module_mt = { - __index = error_on_nil_read, - - __newindex = function() - error("modifying module table forbidden", 2) - end, - - __metatable = true, -} - --- Will contain a function to restore gamevars when running from savegame --- restoration. See SAVEFUNC_ARGS for its arguments. -local g_restorefunc = nil - --- Local gamevar restoration function run from --- our_require('end_gamevars') <- [user module]. -local function restore_local(li, lval) - -- level: - -- 0 is getlocal() itself. - -- 1 is this function (restore_local). - -- 2 is the function calling restore_local(), the savecode. - -- 3 is the function calling the savecode, our_require. - -- 4 is the function calling our_require, the module function. - if (ffiC._DEBUG_LUNATIC ~= 0) then - printf("Restoring index #%d (%s) with value %s", - li, debug.getlocal(4, li), tostring(lval)) - end - - assert(debug.setlocal(4, li, lval)) -end - --- The "require" function accessible to Lunatic code. --- Base modules in allowed_modules are wrapped so that they cannot be --- modified, user modules are searched in the EDuke32 search --- path. Also, our require --- * never messes with the global environment, it only returns the module. --- * allows passing varargs beyond the name to the module. -local function our_require(modname, ...) - local ERRLEV = 5 - - -- Check module name is valid first. - -- TODO: restrict valid names? - if (type(modname) ~= "string" or #modname==0) then - error("module name must be a nonempty string", 2) - end - - -- For _LUNATIC_DBG - if (modname:match("^_LUNATIC") and ffiC._DEBUG_LUNATIC == 0) then - return nil - end - - -- Handle the section between module() and require("end_gamevars"). - if (modname == "end_gamevars") then - local thismodname = getcurmodname("require") - - if (module_gamevars[thismodname] ~= nil) then - error("\"require 'end_gamevars'\" must be called at most once per require'd file", 2) - end - - local gvnames = {} - - for name in pairs(getfenv(2)) do - gvnames[#gvnames+1] = name - if (ffiC._DEBUG_LUNATIC ~= 0) then - printf("MODULE %s GAMEVAR %s", thismodname, name) - end - end - - module_gamevars[thismodname] = gvnames - local gvmodi = module_gvlocali[thismodname] - gvmodi[2] = getnumlocals() - - if (ffiC._DEBUG_LUNATIC ~= 0) then - local numlocals = gvmodi[2]-gvmodi[1]+1 - if (numlocals > 0) then - printf("Module '%s' has %d locals, index %d to %d", - thismodname, numlocals, gvmodi[1], gvmodi[2]) - end - end - - -- Potentially restore gamevars. - if (g_restorefunc) then - local modtab = package_loaded[thismodname] - assert(type(modtab)=="table") - -- SAVEFUNC_ARGS. - g_restorefunc(thismodname, modtab, restore_local) - end - - -- Return whether we're NOT running from a savegame restore in the - -- second outarg. (Lunatic-private!) - return nil, (g_restorefunc==nil) - end - - -- See whether it's a base module name. - if (allowed_modules[modname] ~= nil) then - return allowed_modules[modname] - end - - --- Search user modules... - - if (modname:find("[/\\]")) then - error("Module name must not contain directory separators", ERRLEV-1) - end - -- Instead, dots are translated to directory separators. For EDuke32's - -- virtual file system, this is always a forward slash. Keep the original - -- module name for passing to the module function. - local omodname = modname - modname = modname:gsub("%.", "/") - - local omod = package_loaded[modname] - if (omod ~= nil) then - if (omod==false) then - error("Loop while loading modules", ERRLEV-1) - end - - -- already loaded - assert(omod==true or type(omod)=="table") - return omod - end - - local modfn = modname .. ".lua" - local str = readintostr_mod(modfn) - if (str == nil) then - errorf(ERRLEV-1, "Couldn't open file \"%s\"", modfn) - end - - -- Implant code that yields the module thread just before it would return - -- otherwise. - str = str.."\nrequire('coroutine').yield()" - - local modfunc, errmsg = loadstring(str, modfn) - if (modfunc == nil) then - errorf(ERRLEV-1, "Couldn't load \"%s\": %s", modname, errmsg) - end - - package_loaded[modname] = false -- 'not yet loaded' - table.insert(modname_stack, modname) - - -- Run the module code in a separate Lua thread! - local modthread = coroutine.create(modfunc) - local ok, retval = coroutine.resume(modthread, omodname, ...) - - if (not ok) then - errorf(ERRLEV-1, "Failed running \"%s\": %s\n%s", modname, - retval, debug.traceback(modthread)) - end - - table.remove(modname_stack) - - local modtab = package_loaded[modname] - - if (type(modtab) ~= "table") then - -- The module didn't call our 'module'. Check if it returned a table. - -- In that case, the coroutine has finished its main function and has - -- not reached our implanted 'yield'. - if (coroutine.status(modthread)=="dead" and type(retval)=="table") then - modtab = retval - package_loaded[modname] = modtab - else - package_loaded[modname] = true - end - end - - if (type(modtab) == "table") then - -- Protect module table in any case (i.e. either if the module used our - -- 'module' or if it returned a table). - setmetatable(modtab, required_module_mt) - end - - local gvmodi = module_gvlocali[modname] - - if (gvmodi and gvmodi[2] and gvmodi[2]>=gvmodi[1]) then - if (coroutine.status(modthread)=="suspended") then - -- Save off the suspended thread so that we may get its locals later on. - -- It is never resumed, but only ever used for debug.getlocal(). - module_thread[modname] = modthread - - if (ffiC._DEBUG_LUNATIC ~= 0) then - printf("Keeping coroutine for module \"%s\"", modname) - end - end - end - - return modtab -end - - -local module_mt = { - __index = error_on_nil_read, -} - --- Our 'module' replacement doesn't get the module name from the function args --- since a malicious user could remove other loaded modules this way. --- Also, our 'module' takes no varargs ("option functions" in Lua). --- TODO: make transactional? -local function our_module() - if (#modname_stack == 0) then - error("'module' must be called at the top level of a require'd file", 2) - -- ... as opposed to "at runtime". - end - - local modname = getcurmodname("module") - - if (package_loaded[modname]) then - error("'module' must be called at most once per require'd file", 2) - end - - local M = setmetatable({}, module_mt) - package_loaded[modname] = M - -- change the environment of the function which called us: - setfenv(2, M) - - module_gvlocali[modname] = { getnumlocals()+1 } -end - --- overridden 'error' that always passes a string to the base 'error' -local function our_error(errmsg, level) - if (type(errmsg) ~= "string") then - error("error using 'error': error message must be a string", 2) - end - - if (level) then - if (type(level) ~= "number") then - error("error using 'error': error level must be a number", 2) - end - - error(errmsg, level==0 and 0 or level+1) - end - - error(errmsg, 2) -end - - --- _G tweaks -- pull in only 'safe' stuff -local G_ = {} -- our soon-to-be global environment - -G_.assert = assert -G_.error = our_error -G_.ipairs = ipairs -G_.pairs = pairs -G_.pcall = pcall -G_.print = print -G_.module = our_module -G_.next = next -G_.require = our_require -G_.select = select -G_.tostring = tostring -G_.tonumber = tonumber -G_.type = type -G_.unpack = unpack -G_.xpcall = xpcall -G_._VERSION = _VERSION - --- Available through our 'require': --- bit, coroutine, math, string, table - --- Not available: --- collectgarbage, debug, dofile, gcinfo (DEPRECATED), getfenv, getmetatable, --- jit, load, loadfile, loadstring, newproxy (NOT STD?), package, rawequal, --- rawget, rawset, setfenv, setmetatable - -G_._G = G_ - --- Chain together two functions taking 3 input args. -local function chain_func3(func1, func2) - if (func1==nil or func2==nil) then - return assert(func1 or func2) - end - - -- Return a function that runs first and then tail-calls . - return function(aci, pli, dist) - func1(aci, pli, dist) - return func2(aci, pli, dist) - end -end - --- Determines the last numeric index of a table using *pairs*, so that in --- arg-lists with "holes" (e.g. {1, 2, nil, function() end}) are handled --- properly. -local function ourmaxn(tab) - local maxi = 0 - for i in pairs(tab) do - if (type(i)=="number") then - maxi = math.max(i, maxi) - end - end - assert(tab[maxi] ~= nil) - return maxi -end - --- Running for the very first time? -local g_firstRun = (ffiC.g_elCONSize == 0) - --- Actor functions, saved for actor chaining -local actor_funcs = {} --- Event functions, saved for event chaining -local event_funcs = {} - --- Per-actor sprite animation callbacks -local animsprite_funcs = {} - -local gameactor_internal = gameactor_internal -- included in lunatic.c -local gameevent_internal = gameevent_internal -- included in lunatic.c - -local function animate_all_sprites() - for i=0,ffiC.spritesortcnt-1 do - local tspr = ffiC.tsprite[i] - - if (tspr.owner < ffiC.MAXSPRITES+0ULL) then - local spr = tspr:getspr() - local animfunc = animsprite_funcs[spr.picnum] - - if (animfunc) then - animfunc(tspr) - end - end - end -end - - -local function check_arg_number(name, argpos, val) - if (type(val) ~= "number") then - errorf(3, "invalid '%s' argument (#%d) to gameactor: must be a number", name, argpos) - end -end - --- gameactor{tilenum [, flags [, strength [, action [, move [, movflags]]]]], func} --- Every arg may be positional OR key=val (with the name indicated above as key), --- but not both. -local function our_gameactor(args) - bcheck.top_level("gameactor") - - if (type(args)~="table") then - error("invalid gameactor call: must be passed a table") - end - - local tilenum = args[1] - if (type(tilenum) ~= "number") then - error("invalid argument #1 to gameactor: must be a number", 2) - end - if (not (tilenum >= 0 and tilenum < ffiC.MAXTILES)) then - error("invalid argument #1 to gameactor: must be a tile number [0..gv.MAXTILES-1]", 2) - end - - local lastargi = ourmaxn(args) - local func = args[lastargi] - if (type(func) ~= "function") then - func = args.func - lastargi = 1/0 - end - if (type(func) ~= "function") then - error("invalid gameactor call: must provide a function with last numeric arg or .func", 2) - end - - local flags = (lastargi > 2 and args[2]) or args.flags or 0 - check_arg_number("flags", 2, flags) - - local AF = actor.FLAGS - local chainflags = band(flags, AF._CHAIN_MASK_ACTOR) - flags = band(flags, BNOT.CHAIN_MASK_ACTOR) - - if (chainflags == 0) then - -- Default chaining behavior: don't, replace the old actor instead. - chainflags = AF.replace - elseif (band(chainflags, chainflags-1) ~= 0) then - error("invalid chaining control flags to gameactor", 2) - end - - local replacep = (chainflags==AF.replace) - if (not replacep and not actor_funcs[tilenum]) then - error("attempt to chain code to nonexistent actor tile "..tilenum, 2) - end - - local flags_rbits = band(flags, BNOT.USER_MASK) - if (flags_rbits ~= 0) then - error("invalid 'flags' argument (#2) to gameactor: must not set reserved bits (0x" - ..(bit.tohex(flags_rbits))..")", 2) - end - - local strength = ((lastargi > 3 and args[3]) or args.strength) or (replacep and 0 or nil) - if (replacep or strength~=nil) then - check_arg_number("strength", 3, strength) - end - - local act = ((lastargi > 4 and args[4]) or args.action) or (replacep and literal_act[0] or nil) - if (replacep or act ~= nil) then - if (type(act)=="number" and (act==0 or act==1)) then - act = literal_act[act] - elseif (not ffi.istype(con_action_ct, act)) then - error("invalid 'action' argument (#4) to gameactor: must be an action", 2) - end - end - - local mov = ((lastargi > 5 and args[5]) or args.move) or (replacep and literal_mov[0] or nil) - if (replacep or mov ~= nil) then - if (type(mov)=="number" and (mov==0 or mov==1)) then - mov = literal_mov[mov] - elseif (not ffi.istype(con_move_ct, mov)) then - error("invalid 'move' argument (#5) to gameactor: must be a move", 2) - end - end - - local movflags = ((lastargi > 6 and args[6]) or args.movflags) or (replacep and 0 or nil) - if (replacep or movflags ~= nil) then - check_arg_number("movflags", 6, movflags) - end - - -- Register a potentially passed drawn sprite animation callback function. - -- TODO: allow registering without main actor execution callback. - local animfunc = args.animate - if (animfunc ~= nil) then - if (type(animfunc) ~= "function") then - error("invalid 'animate' argument to gameactor: must be a function", 2) - end - - animsprite_funcs[tilenum] = replacep and func - or (chainflags==AF.chain_beg) and chain_func3(animfunc, animsprite_funcs[tilenum]) - or (chainflags==AF.chain_end) and chain_func3(animsprite_funcs[tilenum], animfunc) - or assert(false) - - -- Register our EVENT_ANIMATEALLSPRITES only now so that it is not - -- called if there are no 'animate' definitions. - gameevent_internal(97, animate_all_sprites) -- EVENT_ANIMATEALLSPRITES - end - - -- All good, bitwise-OR the tile bits and register the actor! - ffiC.g_tile[tilenum]._flags = bit.bor(ffiC.g_tile[tilenum]._flags, flags) - - local newfunc = replacep and func - or (chainflags==AF.chain_beg) and chain_func3(func, actor_funcs[tilenum]) - or (chainflags==AF.chain_end) and chain_func3(actor_funcs[tilenum], func) - or assert(false) - - gameactor_internal(tilenum, strength, act, mov, movflags, newfunc) - actor_funcs[tilenum] = newfunc -end - - --- gameevent{ [, flags], } -local function our_gameevent(args) - bcheck.top_level("gameevent") - - if (type(args)~="table") then - error("invalid gameevent call: must be passed a table") - end - - local event = args[1] - - if (type(event) == "string") then - if (event:sub(1,6) ~= "EVENT_") then - event = "EVENT_"..event - end - local eventidx = con_lang.EVENT[event] - if (eventidx == nil) then - errorf(2, "gameevent: invalid event label %q", event) - end - event = eventidx - end - if (type(event) ~= "number") then - error("invalid argument #1 to gameevent: must be a number or event label", 2) - end - if (not (event >= 0 and event < con_lang.MAXEVENTS)) then - error("invalid argument #1 to gameevent: must be an event number (0 .. MAXEVENTS-1)", 2) - end - - local AF = actor.FLAGS - - -- Kind of CODEDUP from our_gameactor. - local lastargi = ourmaxn(args) - local func = args[lastargi] - if (type(func) ~= "function") then - func = args.func - lastargi = 1/0 - end - if (type(func) ~= "function") then - error("invalid gameevent call: must provide a function with last numeric arg or .func", 2) - end - - -- Event chaining: in Lunatic, chaining at the *end* is the default. - local flags = (lastargi > 2 and args[2]) or args.flags or AF.chain_end - if (type(flags) ~= "number") then - error("invalid 'flags' argument (#2) to gameevent: must be a number", 2) - end - - if (band(flags, BNOT.CHAIN_MASK_EVENT) ~= 0) then - error("invalid 'flags' argument to gameevent: must not set reserved bits", 2) - end - - local newfunc = (flags==AF.replace) and func - or (flags==AF.chain_beg) and chain_func3(func, event_funcs[event]) - or (flags==AF.chain_end) and chain_func3(event_funcs[event], func) - or assert(false) - - gameevent_internal(event, newfunc) - event_funcs[event] = newfunc -end - ---- non-default data and functions -G_.gameevent = our_gameevent -G_.gameactor = our_gameactor --- These come from above: -G_.player = player -G_.actor = actor -G_.projectile = projectile -G_.g_tile = g_tile - -G_.LUNATIC_FIRST_TIME = (ffiC.g_elFirstTime ~= 0) - --- A table that can be used for temporary data when debugging from the OSD. -G_.d = {} - - ----=== Lunatic translator setup ===--- - -read_into_string = readintostr_mod -- for lunacon -local lunacon = require("lunacon") - -local concode, lineinfo - ---- Get Lua code for CON (+ mutator) code. -if (g_firstRun) then - -- Compiling CON for the first time. - local confn = { ffi.string(ffiC.G_ConFile()) } - - local nummods = ffiC.g_scriptModulesNum - if (nummods > 0) then - assert(ffiC.g_scriptModules ~= nil) - - for i=1,nummods do - confn[i+1] = ffi.string(ffiC.g_scriptModules[i-1]) - end - end - - concode, lineinfo = lunacon.compile(confn) - - if (concode == nil) then - error("Failure compiling CON code, exiting.", 0) - end - assert(lineinfo) - - -- Back up the translated code on the C side. - assert(type(concode)=="string") - ffiC.El_SetCON(concode) -else - -- CON was already compiled. - concode = ffi.string(ffiC.g_elCON, ffiC.g_elCONSize) - lineinfo = lunacon.get_lineinfo(concode) -end - -if (ffiC._DEBUG_LUNATIC ~= 0) then - -- XXX: lineinfo of 2nd up time has one line less. - printf("CON line info has %d Lua lines", #lineinfo.llines) -end - -do - -- Translate one Lua line number to a CON file name + line number - local function transline(lnum) - return string.format("%s:%d", lineinfo:getfline(tonumber(lnum))) - end - - -- Register the function that tweaks an error message, looking out for - -- errors from CON and translating the line numbers. - local function tweak_traceback_msg(errmsg) - return errmsg:gsub('%[string "CON"%]:([0-9]+)', transline) - end - - lprivate.tweak_traceback_msg = tweak_traceback_msg - - set_tweak_traceback_internal(tweak_traceback_msg) -end - --- XXX: May still be require'd from user code, we don't want that (at least not --- under this name). -local CON_MODULE_NAME = "_CON\0" - --- Set up Lunatic gamevar serialization. -do - local savegame = require("lunasave") - - -- Callback for: const char *(int32_t *slenptr, int32_t levelnum); - ffiC.El_SerializeGamevars = function(slenptr, levelnum) - local sb = savegame.savebuffer() - - -- Module name, module table, restore_local. See SAVEFUNC_ARGS. - sb:addraw("local N,M,F=...") - -- A local to temporarily hold module locals. - sb:addraw("local L") - - -- XXX: System gamevars? Most of them ought to be saved with C data. - for modname, modvars in pairs(module_gamevars) do - local isCON = (modname==CON_MODULE_NAME) - - sb:startmod(modname) - - -- Handle global gamevars first. - for i=1,#modvars do - local varname = modvars[i] - local excludedVars = isCON and varname=="_V" and - package_loaded[CON_MODULE_NAME]._V._IS_NORESET_GAMEVAR or nil - - -- Serialize gamevar named 'varname' from module named 'modname'. - -- XXX: May error. This will terminate EDuke32 since this callback - -- is run unprotected. - if (sb:add("M."..varname, package_loaded[modname][varname], excludedVars)) then - -- We couldn't serialize that gamevar. - slenptr[0] = -1 - -- Signal which gamevar that was. - return (isCON and "" or modname).."."..varname - end - end - - local modthread = module_thread[modname] - - if (modthread) then - -- Handle local gamevars. - local gvmodi = module_gvlocali[modname] - - for li=gvmodi[1],gvmodi[2] do - -- Serialize local with index

  • . Get its value first. - local lname, lval = debug.getlocal(modthread, 1, li) - - if (sb:add("L", lval)) then - -- We couldn't serialize that gamevar. - slenptr[0] = -1 - return "local "..modname.."."..lname - end - - -- Emit code to call restore_local. - sb:addrawf("F(%d,L)", li) - end - end - - sb:endmod() - end - - -- Get the whole code as a string. - local savecode = sb:getcode() - - if (ffiC._DEBUG_LUNATIC ~= 0) then - -- Dump the code if Lunatic debugging is enabled (-Lopts=diag) and - -- there is a LUNATIC_SAVECODE_FN variable in the environment. - local os = require("os") - local fn = os.getenv("LUNATIC_SAVECODE_FN") - - if (fn ~= nil) then - if (levelnum >= 0) then - fn = fn .. levelnum - end - - local io = require("io") - local f = io.open(fn, "w") - - if (f ~= nil) then - f:write(savecode) - f:close() - printf("Wrote Lunatic gamevar restoration code to \"%s\".", fn) - end - end - end - - -- Set the size of the code and return the code to C. - slenptr[0] = #savecode - return savecode - end -end - --- change the environment of this chunk to the table G_ --- NOTE: all references to global variables from this point on --- (also in functions created after this point) refer to G_ ! -setfenv(1, G_) - --- Print keys and values of a table. -local function printkv(label, table) - print("========== Keys and values of "..label.." ("..tostring(table)..")") - for k,v in pairs(table) do - print(k .. ": " .. tostring(v)) - end - print("----------") -end - ---printkv('_G AFTER SETFENV', _G) - - ----=== Restricted access to C variables from Lunatic ===--- - --- error(..., 2) is to blame the caller and get its line numbers - --- Map of 'gv' variable names to C ones. -local varnameMap = { - gametic = "g_moveThingsCount", - RETURN = "g_RETURN", - _sessionVar = "g_elSessionVar", -} - -gv_access.gametic = true -gv_access.RETURN = true -gv_access._sessionVar = true - -local tmpmt = { - __index = function(_, key) - if (gv_access[key] == nil) then - -- Read access allowed. - return ffiC[key] - elseif (type(gv_access[key])~="boolean") then - -- Overridden 'gv' pseudo-member... - return gv_access[key] - elseif (varnameMap[key]) then - -- Variable known under a different name on the C side. - return ffiC[varnameMap[key]] - end - error("access forbidden", 2) - end, - - __newindex = function(_, key, val) - if (gv_access[key] == nil) then - -- Variables declared 'const' are handled by LuaJIT. - ffiC[key] = val - elseif (varnameMap[key]) then - ffiC[varnameMap[key]] = val - else - error("write access forbidden", 2) - end - end, -} -gv = setmtonce(gv_, tmpmt) - --- This will create 'sprite', 'wall', etc. HERE, i.e. in the environment of this chunk -defs_c.create_globals(_G) - --- REMOVE this for release -if (ffiC._DEBUG_LUNATIC ~= 0) then - local DBG_ = {} - DBG_.debug = require("debug") - DBG_.printkv = printkv - DBG_.loadstring = loadstring - DBG_.oom = function() - local s = "1" - for i=1,math.huge do - s = s..s - end - end - - -- Test reading from all struct members - DBG_.testmembread = function() - for _1, sac in pairs { con_lang.StructAccessCode, con_lang.StructAccessCode2 } do - for what, labels in pairs(sac) do - if (what~="tspr") then - for _3, membaccode in pairs(labels) do - if (type(membaccode)=="table") then - membaccode = membaccode[1] - end - if (membaccode) then - local codestr = [[ -do - local _bit,_math=require'bit',require'math' - local _con=require'con' - local _band,_gv=_bit.band,gv -local tmp=]].. - membaccode:gsub("^_gud%(_pli%)", "_con._get_userdef(0)"):gsub("%%s","0").." end" - local code = assert(loadstring(codestr)) - code() - end - end - end - end - end - end - - allowed_modules._LUNATIC_DBG = DBG_ -end - ----=== Finishing environment setup ===--- - ---printkv('_G AFTER DECLS', _G) - -local index_error_mt = { - __index = function(_, key) - error("attempt to read undeclared variable '"..key.."'", 2) - end, - - __metatable = true, -} - --- PiL 14.2 continued --- We need this at the end because we were previously doing just that! -setmetatable(G_, index_error_mt) - -local global_mt = { - __index = G_, - - __newindex = function() - error("attempt to write into the global environment") - end, - - __metatable = true, -} - --- Change the environment of the running Lua thread so that everything --- what we've set up will be available when this chunk is left. --- In particular, we need the globals defined after setting this chunk's --- environment earlier. -setfenv(0, setmetatable({}, global_mt)) - -do - -- If we're running from a savegame restoration, create the restoration - -- function. Must be here, after the above setfenv(), because it must be - -- created in this protected ('user') context! - local cstr = ffiC.g_elSavecode - if (cstr~=nil) then - local restorecode = ffi.string(cstr) - ffiC.El_FreeSaveCode() - - g_restorefunc = assert(loadstring(restorecode)) - end -end - --- Restore CON gamevars from loadmapstate. --- TODO: non-user-defined gamevars. --- TODO: savegames. --- int32_t El_RestoreGamevars(const char *) -ffiC.El_RestoreGamevars = function(savecode) - savecode = ffi.string(savecode) - local restorefunc = assert(loadstring(savecode)) - restorefunc(CON_MODULE_NAME, package_loaded[CON_MODULE_NAME], nil) - return 0 -end - --- Run the CON code translated into Lua. -if (assert(concode ~= nil)) then - local confunc, conerrmsg = loadstring(concode, "CON") - if (confunc == nil) then - error("Failure loading translated CON code: "..conerrmsg, 0) - end - - -- Emulate our 'require' for the CON module when running it, for - -- our_module() which is called from the generated Lua code. - table.insert(modname_stack, CON_MODULE_NAME) - local conlabels, conaction, conmove, conai = confunc() - table.remove(modname_stack) - - local function protect_con_table(tab) - -- NOTE: Some of our code specifically excepts the name tables to be - -- indexable with nonexistent keys. See e.g. control.lua: _A_SpawnGlass() - if (ffiC._LUNATIC_STRICT ~= 0) then - tab = setmetatable(tab, index_error_mt) - end - return setmtonce({}, { __index=tab, __newindex=basemod_newidx }) - end - - -- Set up CON.* modules, providing access to diffenrent kinds of labels - -- defined in CON from Lua. See CONCODE_RETURN in lunacon.lua. - allowed_modules["CON.DEFS"] = protect_con_table(conlabels) - allowed_modules["CON.ACTION"] = protect_con_table(conaction) - allowed_modules["CON.MOVE"] = protect_con_table(conmove) - allowed_modules["CON.AI"] = protect_con_table(conai) - - -- Propagate potentially remapped defines to the control module. - con._setuplabels(conlabels) - - local function getLabelValue(str, doupper) - return conlabels[doupper and string.upper(str) or str] - end - - ffiC.El_GetLabelValue = function(label) - local str = ffi.string(label) - local ok, val = pcall(getLabelValue, str, false) - if (not ok or type(val)~="number") then - ok, val = pcall(getLabelValue, str, true) - end - return (ok and type(val)=="number") and val or bit.tobit(0x80000000) - end -end - --- When starting a map, load Lua modules given on the command line. -if (not g_firstRun) then - local i=0 - while (ffiC.g_elModules[i] ~= nil) do - -- Get the module name and strip the trailing extension. - local modname = ffi.string(ffiC.g_elModules[i]):gsub("%.lua$","") - - if (modname:find("%.")) then - -- Because they will be replaced by dirseps in our_require(). - error("Dots are not permitted in module names", 0) - end - -- Allow forward slashes in module names from the cmdline. - our_require((modname:gsub("%.lua$",""):gsub("/","."))) - - i = i+1 - end - - ffiC.g_elFirstTime = 0 -end - -if (g_restorefunc) then - -- Clear it so that it may be garbage-collected. - g_restorefunc = nil -end diff --git a/source/duke3d/src/lunatic/bcarray.lua b/source/duke3d/src/lunatic/bcarray.lua deleted file mode 100644 index 6d78e33db..000000000 --- a/source/duke3d/src/lunatic/bcarray.lua +++ /dev/null @@ -1,124 +0,0 @@ --- Implementation of a bound-checked array type factory for LuaJIT 2.0 or later. --- --- Usage example: --- --- > bcarray.new("int8_t", 3, "test", "three_pigs") --- > a = ffi.new("struct { int32_t a; three_pigs p; int16_t b; }") --- > =ffi.sizeof(a) --> 12 --- > b = ffi.new("__attribute__((packed)) struct { int32_t a; three_pigs p; int16_t b; }") --- > =ffi.sizeof(b) --> 9 - -local ffi = require("ffi") - -local string = require("string") -local table = require("table") - -local assert = assert -local error = error -local pairs = pairs -local type = type - - -local bcarray = {} - - --- Generate C decl for a sequence of const struct members. --- For example, for 4 elements, --- "const $ _r1, _f2, _u3, _n4;" -local function flatten_array(nelts, rng) - local strtab = { "$ " } - - if (rng and rng.getu32==nil) then - assert(type(rng)=="table") - - for i=1,#rng do - strtab[i+1] = rng[i]..((i<#rng) and "," or ";") - end - else - for i=1,nelts do - local ch = 97 + (rng and (rng:getu32() % 25) or 0) -- 'a'..'z' - strtab[i+1] = string.format("_%c%x%s", ch, i, (i: Number of elements in array (small number) --- : The name to be shown for the derived type in error messages --- : If non-nil, the name under which the derived type is typedef'd --- : Random generator state + method :getu32(). If nil, then members are --- named _a1, _a2, ... --- It also may be a table containing member names at numeric indices 1..#rng. --- : A table containing functions __index and/or __newindex. They are --- called first and the bound-checking ones are tail-called then. If the --- custom __index one returns something, it is returned by the composite one. -function bcarray.new(basetype, numelts, showname, typename, rng, mtadd) - local eltptr_t = ffi.typeof("$ *", ffi.typeof(basetype)) - - local mt = { - __index = function(ar, idx) - if (not (idx >= 0 and idx < numelts)) then - error("out-of-bounds "..showname.." read access", 2) - end - return ffi.cast(eltptr_t, ar)[idx] - end, - - -- NOTE: this function will be dead code if the prefixed __newindex - -- errors out unconditionally or the bcarray is declared 'const'. - __newindex = function(ar, idx, val) - if (not (idx >= 0 and idx < numelts)) then - error("out-of-bounds "..showname.." write access", 2) - end - ffi.cast(eltptr_t, ar)[idx] = val - end, - } - - if (mtadd ~= nil) then - local curindexf, curnewindexf = mt.__index, mt.__newindex - local addindexf, addnewindexf = mtadd.__index, mtadd.__newindex - - if (addindexf) then - -- Additional __index metamethod given. - mt.__index = function(ar, idx) - local sth = addindexf(ar, idx) - if (sth ~= nil) then - return sth - end - return curindexf(ar, idx) - end - end - - if (addnewindexf) then - -- Additional __newindex metamethod given. - mt.__newindex = function(ar, idx, val) - addnewindexf(ar, idx, val) - return curnewindexf(ar, idx, val) - end - end - end - - local cdeclstr = "struct {"..flatten_array(numelts, rng).."}" - local bcarray_t = ffi.typeof(cdeclstr, ffi.typeof(basetype)); - - bcarray_t = ffi.metatype(bcarray_t, mt) - if (not (rng and rng.getu32==nil)) then - -- When passing a member name table, it is allowed to have a different - -- number of named members than array elements. - assert(ffi.sizeof(bcarray_t) == ffi.sizeof(basetype)*numelts) - end - - if (typename ~= nil) then - -- Register the type name in the global namespace. - assert(type(typename)=="string") - ffi.cdef("typedef $ $;", bcarray_t, typename) - end - - return bcarray_t -end - - -return bcarray diff --git a/source/duke3d/src/lunatic/bcheck.lua b/source/duke3d/src/lunatic/bcheck.lua deleted file mode 100644 index 55690df45..000000000 --- a/source/duke3d/src/lunatic/bcheck.lua +++ /dev/null @@ -1,126 +0,0 @@ --- Bound-checking functions for engine and game "things". - -local ffiC = require("ffi").C -local type = type -local error = error - -local bcheck = {} - ---== ENGINE ==-- - -function bcheck.sector_idx(sectnum) - if (not (sectnum >= 0 and sectnum < ffiC.numsectors)) then - error("invalid sector number "..sectnum, 3) - end -end - -function bcheck.wall_idx(wallnum) - if (not (wallnum >= 0 and wallnum < ffiC.numwalls)) then - error("invalid wall number "..wallnum, 3) - end -end - --- TODO: Provide another function that also checks whether the sprite exists in --- the game world (statnum != MAXSTATUS). -function bcheck.sprite_idx(spritenum) --- if (not (spritenum >= 0 and spritenum < ffiC.MAXSPRITES)) then - if (not (spritenum >= 0 and spritenum < ffiC.MAXSPRITES)) then - error("invalid sprite number "..spritenum, 3) - end -end - -function bcheck.tile_idx(tilenum) - if (not (tilenum >= 0 and tilenum < ffiC.MAXTILES)) then - error("invalid tile number "..tilenum, 3) - end -end - - ---== HELPERS ==-- - -function bcheck.number(val, errlev) - if (type(val)~="number" or val~=val) then - error("invalid argument: must be a non-NaN number", errlev or 3) - end -end - -function bcheck.type(val, typestr, errlev) - if (type(val)~=typestr) then - error("invalid argument: must be a "..typestr, errlev or 3) - end -end - - ---== GAME ==-- -if (ffiC.LUNATIC_CLIENT == ffiC.LUNATIC_CLIENT_MAPSTER32) then - return bcheck -end - -local con_lang = require("con_lang") - -function bcheck.player_idx(snum) - if (not (snum >= 0 and snum < ffiC.g_mostConcurrentPlayers)) then - error("invalid player number "..snum, 3) - end -end - -function bcheck.sound_idx(sndidx) - if (not (sndidx >= 0 and sndidx < con_lang.MAXSOUNDS)) then - error("invalid sound number "..sndidx, 3) - end -end - -function bcheck.weapon_idx(weap) - if (not (weap >= 0 and weap < ffiC.MAX_WEAPONS)) then - error("Invalid weapon ID "..weap, 3) - end -end - -function bcheck.inventory_idx(inv) - if (not (inv >= 0 and inv < ffiC.GET_MAX)) then - error("Invalid inventory ID "..inv, 3) - end -end - -function bcheck.volume_idx(volume) - if (not (volume >= 0 and volume < con_lang.MAXVOLUMES)) then - error("invalid volume number "..volume, 3) - end -end - -function bcheck.level_idx(level) - if (not (level >= 0 and level < con_lang.MAXLEVELS)) then - error("invalid level number "..level, 3) - end -end - -function bcheck.linear_map_idx(idx) - if (not (idx >= 0 and idx <= con_lang.MAXLEVELS * con_lang.MAXVOLUMES)) then - error("invalid linear map index "..idx, 3) - end -end - -function bcheck.quote_idx(qnum, onlyidx) - if (not (qnum >= 0 and qnum < con_lang.MAXQUOTES)) then - error("invalid quote number "..qnum, 3) - end - - local cstr = ffiC.apStrings[qnum] - if (onlyidx) then - return cstr - end - - if (cstr == nil) then - error("null quote "..qnum, 3) - end - - return cstr -end - -function bcheck.top_level(funcname, errlev) - if (ffiC.g_elCallDepth > 0) then - error("Invalid use of "..funcname..": must be called from top level", errlev or 3) - end -end - -return bcheck diff --git a/source/duke3d/src/lunatic/bitar.lua b/source/duke3d/src/lunatic/bitar.lua deleted file mode 100644 index f429a8bd8..000000000 --- a/source/duke3d/src/lunatic/bitar.lua +++ /dev/null @@ -1,293 +0,0 @@ - --- "Bit array" module based on LuaJIT's BitOp. - -local bit = require "bit" -local math = require "math" - -local ffi = require "ffi" - -local assert = assert -local error = error -local type = type - -local tostring = tostring - - -module(...) - - -local bitar_ct = ffi.typeof[[ -struct { - const double maxbidx; // last permissible bit index - const double maxidx; // last permissible int32_t index - const intptr_t arptr; // address of the int32_t array -}]] -local ptr_to_int = ffi.typeof("int32_t *") - -local anchor = {} - --- population count of a nibble -local nibpop = ffi.new("double [?]", 16, - { 0, 1, 1, 2, 1, 2, 2, 3, - 1, 2, 2, 3, 2, 3, 3, 4 }) --- ...and of a byte -local bytepop = ffi.new("double [?]", 256) -for i=0,255 do - bytepop[i] = nibpop[bit.band(i, 15)] + nibpop[bit.rshift(i, 4)] -end -nibpop = nil - -local function bitar_from_intar(maxbidx, maxidx, ar) - -- We need to have the int32_t[?] array be reachable so that it will not be - -- garbage collected - local ar_intptr = ffi.cast("intptr_t", ar) - anchor[tostring(ar_intptr)] = ar - - -- Leaving the (potential) high trailing bits at 0 lets us not worry - -- about them in the population count calculation (__len metamethod). - -- Also, this is correct for maxbidx%32 == 0, since BitOp's shifts - -- mask the 5 lower bits of the counts. - local numremain = bit.band(maxbidx+1, 31) - ar[maxidx] = bit.band(ar[maxidx], bit.rshift(-1, 32-numremain)) - - return bitar_ct(maxbidx, maxidx, ar_intptr) -end - -local function setop_common_rel(s1, s2) - if (s1.maxbidx ~= s2.maxbidx) then - error("bad arguments to bit array set op: must be of same length", 4) - end - - local ar1 = ffi.cast(ptr_to_int, s1.arptr) - local ar2 = ffi.cast(ptr_to_int, s2.arptr) - - return ar1, ar2 -end - -local function setop_common(s1, s2) - if (not ffi.istype(bitar_ct, s1) or not ffi.istype(bitar_ct, s2)) then - error("bad arguments to bit array set op: both must be 'bitar' types", 3) - end - - local ar1, ar2 = setop_common_rel(s1, s2) - local ar = ffi.new("int32_t [?]", s1.maxidx+1) - - return ar, ar1, ar2 -end - -local mt = { - --- Operational methods - - __add = function(s1, s2) -- set union - local ar, ar1, ar2 = setop_common(s1, s2) - for i=0,s1.maxidx do - ar[i] = bit.bor(ar1[i], ar2[i]) - end - return bitar_from_intar(s1.maxbidx, s1.maxidx, ar) - end, - - __mul = function(s1, s2) -- set intersection - local ar, ar1, ar2 = setop_common(s1, s2) - for i=0,s1.maxidx do - ar[i] = bit.band(ar1[i], ar2[i]) - end - return bitar_from_intar(s1.maxbidx, s1.maxidx, ar) - end, - - __sub = function(s1, s2) -- set difference - local ar, ar1, ar2 = setop_common(s1, s2) - for i=0,s1.maxidx do - ar[i] = bit.band(ar1[i], bit.bnot(ar2[i])) - end - return bitar_from_intar(s1.maxbidx, s1.maxidx, ar) - end, - - __unm = function(s) -- bitwise NOT - local newar = ffi.new("int32_t [?]", s.maxidx+1) - local oldar = ffi.cast(ptr_to_int, s.arptr) - for i=0,s.maxidx do - newar[i] = bit.bnot(oldar[i]) - end - return bitar_from_intar(s.maxbidx, s.maxidx, newar) - end, - - - --- Additional operations - - __index = { - -- TODO: Rename to 'testi', 'seti', 'cleari'; add 'flipi'? - - -- Is bit i set? - isset = function(s, i) - if (not (i >= 0 and i<=s.maxbidx)) then - error("bad bit index for isset: must be in [0.."..s.maxbidx.."]", 2) - end - s = ffi.cast(ptr_to_int, s.arptr) - return (bit.band(s[bit.rshift(i, 5)], bit.lshift(1, i)) ~= 0) - end, - - -- Clear bit i. - set0 = function(s, i) - if (not (i >= 0 and i<=s.maxbidx)) then - error("bad bit index for set0: must be in [0.."..s.maxbidx.."]", 2) - end - s = ffi.cast(ptr_to_int, s.arptr) - local jx = bit.rshift(i, 5) - s[jx] = bit.band(s[jx], bit.rol(0xfffffffe, i)) - end, - - -- Set bit i. - set1 = function(s, i) - if (not (i >= 0 and i<=s.maxbidx)) then - error("bad bit index for set1: must be in [0.."..s.maxbidx.."]", 2) - end - s = ffi.cast(ptr_to_int, s.arptr) - local jx = bit.rshift(i, 5) - s[jx] = bit.bor(s[jx], bit.rol(0x00000001, i)) - end - }, - - - --- Relational methods - - __eq = function(s1, s2) -- set identity - local ar1, ar2 = setop_common_rel(s1, s2) - for i=0,s1.maxidx do - if (bit.bxor(ar1[i], ar2[i]) ~= 0) then - return false - end - end - return true - end, - - __le = function(s1, s2) - local ar1, ar2 = setop_common_rel(s1, s2) - for i=0,s1.maxidx do - if (bit.band(ar1[i], bit.bnot(ar2[i])) ~= 0) then - return false - end - end - return true - end, - - __lt = function(s1, s2) - return s1 <= s2 and not (s2 == s1) - end, - - -- The length operator gets the population count of the bit array, i.e. the - -- number of set bits. - __len = function(s) - local ar = ffi.cast(ptr_to_int, s.arptr) - local popcnt = 0 - for i=0,s.maxidx do - popcnt = popcnt + bytepop[bit.band(ar[i], 255)] + - bytepop[bit.band(bit.rshift(ar[i], 8), 255)] + - bytepop[bit.band(bit.rshift(ar[i], 16), 255)] + - bytepop[bit.rshift(ar[i], 24)] - end - return popcnt - end, - - -- serialization - __tostring = function(s) - local size=s.maxidx+1 - local ar = ffi.cast(ptr_to_int, s.arptr) - - local hdr = "bitar.new("..s.maxbidx..", '" - local ofs = #hdr - local totalstrlen = ofs+8*size+2 - local str = ffi.new("char [?]", totalstrlen) - - ffi.copy(str, hdr, ofs) - - for i=0,s.maxidx do - -- 'a' is ASCII 97 - for nib=0,7 do - str[ofs + 8*i + nib] = 97 + bit.band(bit.rshift(ar[i], 4*nib), 0x0000000f) - end - end - - ffi.copy(str+totalstrlen-2, "')", 2) - - return ffi.string(str, totalstrlen) - end, - - -- On garbage collection of the bitar, clear the array's anchor so that it - -- can be collected too. - __gc = function(s) - anchor[tostring(s.arptr)] = nil - end, - - __metatable = true, -} - -ffi.metatype(bitar_ct, mt) - - --- Create new bit array. -function new(maxbidx, initval) - if (type(maxbidx) ~= "number" or not (maxbidx >= 0 and maxbidx <= (2^31)-2)) then - -- NOTE: Uh-oh, we can't write '2^31' because that would be interpreted - -- as color by OSD_Printf. - error("bad argument #1 to bitar.new (must be a number in [0..(2**31)-2])", 2) - end - - if (math.floor(maxbidx) ~= maxbidx) then - error("bad argument #1 to bitar.new (must be an integral number)") - end - - if (ffi.istype(ptr_to_int, initval)) then - -- Initialization from an array on the C side. INTERNAL. - -- Cannot be reached by user code since there's no access to the FFI - -- (and thus no way to create pointers). - - return bitar_from_intar(maxbidx, (maxbidx+1)/32-1, initval) - - elseif (type(initval)=="string") then - -- String containing hex digits (a..p) given, for INTERNAL use only. - -- XXX: Can be reached by user code. - - local lstr = initval - - local numnibs = #lstr - assert(numnibs%8 == 0) - - local size = numnibs/8 - local maxidx = size-1 - local ar = ffi.new("int32_t [?]", size) - - local str = ffi.new("char [?]", numnibs) - ffi.copy(str, lstr, numnibs) - - for i=0,maxidx do - ar[i] = 0 - - for nib=0,7 do - local hexdig = str[8*i + nib] - assert(hexdig >= 97 and hexdig < 97+16) - ar[i] = bit.bor(ar[i], bit.lshift(hexdig-97, 4*nib)) - end - end - - -- NOTE: cannot be extracted from the string, use the passed one. - return bitar_from_intar(maxbidx, maxidx, ar) - - else - -- User-requested bitar creation. - - if (initval ~= 0 and initval ~= 1) then - error("bad argument #2 to bitar.new (must be either 0 or 1)", 2) - end - - local maxidx = math.floor(maxbidx/32) - local size = maxidx+1 - - local ar = ffi.new("int32_t [?]", size) - - if (initval==1) then - ffi.fill(ar, size*4, -1) - end - - return bitar_from_intar(maxbidx, maxidx, ar) - end -end diff --git a/source/duke3d/src/lunatic/con_lang.lua b/source/duke3d/src/lunatic/con_lang.lua deleted file mode 100644 index 09c1de09d..000000000 --- a/source/duke3d/src/lunatic/con_lang.lua +++ /dev/null @@ -1,1341 +0,0 @@ --- CON language definitions - -local lpeg = require("lpeg") - -local pairs = pairs -local print = print -local setmetatable = setmetatable -local type = type - - -module(...) - - -MAXVOLUMES = 7 -MAXLEVELS = 64 -MAXGAMETYPES = 16 - -MAXSKILLS = 7 - -MAXSOUNDS = 4096 - -MAXSESSIONVARS = 8 -- KEEPINSYNC lunatic_game.c - --- KEEPINSYNC quotes.h - --- For Lunatic, MAXQUOTES is OBITQUOTEINDEX because starting from that index --- are obituary and suicide quotes which are passed as *format strings* to --- sprintf() in C. -REALMAXQUOTES = 16384 -MAXQUOTES = REALMAXQUOTES-128 -MAXQUOTELEN = 128 - -local STR = { - STR_MAPNAME = 0, - STR_MAPFILENAME = 1, - STR_PLAYERNAME = 2, - STR_VERSION = 3, - STR_GAMETYPE = 4, - STR_VOLUMENAME = 5, - STR_YOURTIME = 6, - STR_PARTIME = 7, - STR_DESIGNERTIME = 8, - STR_BESTTIME = 9, -} - -PROJ = { - PROJ_WORKSLIKE = 0, - PROJ_SPAWNS = 1, - PROJ_SXREPEAT = 2, - PROJ_SYREPEAT = 3, - PROJ_SOUND = 4, - PROJ_ISOUND = 5, - PROJ_VEL = 6, - PROJ_EXTRA = 7, - PROJ_DECAL = 8, - PROJ_TRAIL = 9, - PROJ_TXREPEAT = 10, - PROJ_TYREPEAT = 11, - PROJ_TOFFSET = 12, - PROJ_TNUM = 13, - PROJ_DROP = 14, - PROJ_CSTAT = 15, - PROJ_CLIPDIST = 16, - PROJ_SHADE = 17, - PROJ_XREPEAT = 18, - PROJ_YREPEAT = 19, - PROJ_PAL = 20, - PROJ_EXTRA_RAND = 21, - PROJ_HITRADIUS = 22, - PROJ_VEL_MULT = 23, -- NAME (PROJ_MOVECNT) - PROJ_OFFSET = 24, - PROJ_BOUNCES = 25, - PROJ_BSOUND = 26, - PROJ_RANGE = 27, - PROJ_FLASH_COLOR = 28, - PROJ_USERDATA = 29, -} - -EVENT = { - EVENT_INIT = 0, - EVENT_ENTERLEVEL = 1, - EVENT_RESETWEAPONS = 2, - EVENT_RESETINVENTORY = 3, - EVENT_HOLSTER = 4, - EVENT_LOOKLEFT = 5, - EVENT_LOOKRIGHT = 6, - EVENT_SOARUP = 7, - EVENT_SOARDOWN = 8, - EVENT_CROUCH = 9, - EVENT_JUMP = 10, - EVENT_RETURNTOCENTER = 11, - EVENT_LOOKUP = 12, - EVENT_LOOKDOWN = 13, - EVENT_AIMUP = 14, - EVENT_FIRE = 15, - EVENT_CHANGEWEAPON = 16, - EVENT_GETSHOTRANGE = 17, - EVENT_GETAUTOAIMANGLE = 18, - EVENT_GETLOADTILE = 19, - EVENT_CHEATGETSTEROIDS = 20, - EVENT_CHEATGETHEAT = 21, - EVENT_CHEATGETBOOT = 22, - EVENT_CHEATGETSHIELD = 23, - EVENT_CHEATGETSCUBA = 24, - EVENT_CHEATGETHOLODUKE = 25, - EVENT_CHEATGETJETPACK = 26, - EVENT_CHEATGETFIRSTAID = 27, - EVENT_QUICKKICK = 28, - EVENT_INVENTORY = 29, - EVENT_USENIGHTVISION = 30, - EVENT_USESTEROIDS = 31, - EVENT_INVENTORYLEFT = 32, - EVENT_INVENTORYRIGHT = 33, - EVENT_HOLODUKEON = 34, - EVENT_HOLODUKEOFF = 35, - EVENT_USEMEDKIT = 36, - EVENT_USEJETPACK = 37, - EVENT_TURNAROUND = 38, - EVENT_DISPLAYWEAPON = 39, - EVENT_FIREWEAPON = 40, - EVENT_SELECTWEAPON = 41, - EVENT_MOVEFORWARD = 42, - EVENT_MOVEBACKWARD = 43, - EVENT_TURNLEFT = 44, - EVENT_TURNRIGHT = 45, - EVENT_STRAFELEFT = 46, - EVENT_STRAFERIGHT = 47, - EVENT_WEAPKEY1 = 48, - EVENT_WEAPKEY2 = 49, - EVENT_WEAPKEY3 = 50, - EVENT_WEAPKEY4 = 51, - EVENT_WEAPKEY5 = 52, - EVENT_WEAPKEY6 = 53, - EVENT_WEAPKEY7 = 54, - EVENT_WEAPKEY8 = 55, - EVENT_WEAPKEY9 = 56, - EVENT_WEAPKEY10 = 57, - EVENT_DRAWWEAPON = 58, - EVENT_DISPLAYCROSSHAIR = 59, - EVENT_DISPLAYREST = 60, - EVENT_DISPLAYSBAR = 61, - EVENT_RESETPLAYER = 62, - EVENT_INCURDAMAGE = 63, - EVENT_AIMDOWN = 64, - EVENT_GAME = 65, - EVENT_PREVIOUSWEAPON = 66, - EVENT_NEXTWEAPON = 67, - EVENT_SWIMUP = 68, - EVENT_SWIMDOWN = 69, - EVENT_GETMENUTILE = 70, - EVENT_SPAWN = 71, - EVENT_LOGO = 72, - EVENT_EGS = 73, - EVENT_DOFIRE = 74, - EVENT_PRESSEDFIRE = 75, - EVENT_USE = 76, - EVENT_PROCESSINPUT = 77, - EVENT_FAKEDOMOVETHINGS = 78, - EVENT_DISPLAYROOMS = 79, - EVENT_KILLIT = 80, - EVENT_LOADACTOR = 81, - EVENT_DISPLAYBONUSSCREEN = 82, - EVENT_DISPLAYMENU = 83, - EVENT_DISPLAYMENUREST = 84, - EVENT_DISPLAYLOADINGSCREEN = 85, - EVENT_ANIMATESPRITES = 86, - EVENT_NEWGAME = 87, - EVENT_SOUND = 88, - EVENT_CHECKTOUCHDAMAGE = 89, - EVENT_CHECKFLOORDAMAGE = 90, - EVENT_LOADGAME = 91, - EVENT_SAVEGAME = 92, - EVENT_PREGAME = 93, - EVENT_CHANGEMENU = 94, - EVENT_DAMAGEHPLANE = 95, - EVENT_ACTIVATECHEAT = 96, - EVENT_DISPLAYINACTIVEMENU = 97, - EVENT_DISPLAYINACTIVEMENUREST = 98, - EVENT_CUTSCENE = 99, - EVENT_DISPLAYCURSOR = 100, - EVENT_DISPLAYLEVELSTATS = 101, - EVENT_DISPLAYCAMERAOSD = 102, - EVENT_DISPLAYROOMSCAMERA = 103, - EVENT_DISPLAYSTART = 104, - EVENT_WORLD = 105, - EVENT_PREWORLD = 106, - EVENT_PRELEVEL = 107, - EVENT_DISPLAYSPIT = 108, - EVENT_DISPLAYFIST = 109, - EVENT_DISPLAYKNEE = 110, - EVENT_DISPLAYKNUCKLES = 111, - EVENT_DISPLAYSCUBA = 112, - EVENT_DISPLAYTIP = 113, - EVENT_DISPLAYACCESS = 114, - EVENT_RESETGOTPICS = 115, --- EVENT_ANIMATEALLSPRITES = previous+1, -- internal --- KEEPINSYNC with MAXEVENTS below -} - -MAXEVENTS = 116 -- KEEPINSYNC with above EVENT_* list - --- NOTE: negated values are not exported to the ffi.C namespace or CON. --- See TWEAK_SFLAG below. -SFLAG = { - SFLAG_SHADOW = 0x00000001, - SFLAG_NVG = 0x00000002, - SFLAG_NOSHADE = 0x00000004, - SFLAG_PROJECTILE = -0x00000008, - SFLAG_DECAL = -0x00000010, - SFLAG_BADGUY = 0x00000020, - SFLAG_NOPAL = 0x00000040, - SFLAG_NOEVENTS = 0x00000080, -- NAME - SFLAG_NOLIGHT = 0x00000100, - SFLAG_USEACTIVATOR = 0x00000200, - SFLAG_NULL = -0x00000400, - SFLAG_NOCLIP = 0x00000800, - SFLAG_NOFLOORSHADOW = -0x00001000, - SFLAG_SMOOTHMOVE = 0x00002000, - SFLAG_NOTELEPORT = 0x00004000, - SFLAG_BADGUYSTAYPUT = -0x00008000, - SFLAG_CACHE = -0x00010000, - SFLAG_ROTFIXED = -0x00020000, - SFLAG_HARDCODED_BADGUY = -0x00040000, - SFLAG_DIDNOSE7WATER = -0x00080000, - SFLAG_NODAMAGEPUSH = 0x00100000, - SFLAG_NOWATERDIP = 0x00200000, - SFLAG_HURTSPAWNBLOOD = 0x00400000, - -- RESERVED for actor.FLAGS.chain_*/replace_*: - -- 0x08000000, 0x10000000, 0x20000000, 0x40000000 -} - -STAT = { - STAT_DEFAULT = 0, - STAT_ACTOR = 1, - STAT_ZOMBIEACTOR = 2, - STAT_EFFECTOR = 3, - STAT_PROJECTILE = 4, - STAT_MISC = 5, - STAT_STANDABLE = 6, - STAT_LOCATOR = 7, - STAT_ACTIVATOR = 8, - STAT_TRANSPORT = 9, - STAT_PLAYER = 10, - STAT_FX = 11, - STAT_FALLER = 12, - STAT_DUMMYPLAYER = 13, - STAT_LIGHT = 14, --- STAT_NETALLOC = 1023, -- MAXSTATUS-1 -} - -local GAMEFUNC = { - GAMEFUNC_MOVE_FORWARD = 0, - GAMEFUNC_MOVE_BACKWARD = 1, - GAMEFUNC_TURN_LEFT = 2, - GAMEFUNC_TURN_RIGHT = 3, - GAMEFUNC_STRAFE = 4, - GAMEFUNC_FIRE = 5, - GAMEFUNC_OPEN = 6, - GAMEFUNC_RUN = 7, - GAMEFUNC_AUTORUN = 8, - GAMEFUNC_JUMP = 9, - GAMEFUNC_CROUCH = 10, - GAMEFUNC_LOOK_UP = 11, - GAMEFUNC_LOOK_DOWN = 12, - GAMEFUNC_LOOK_LEFT = 13, - GAMEFUNC_LOOK_RIGHT = 14, - GAMEFUNC_STRAFE_LEFT = 15, - GAMEFUNC_STRAFE_RIGHT = 16, - GAMEFUNC_AIM_UP = 17, - GAMEFUNC_AIM_DOWN = 18, - GAMEFUNC_WEAPON_1 = 19, - GAMEFUNC_WEAPON_2 = 20, - GAMEFUNC_WEAPON_3 = 21, - GAMEFUNC_WEAPON_4 = 22, - GAMEFUNC_WEAPON_5 = 23, - GAMEFUNC_WEAPON_6 = 24, - GAMEFUNC_WEAPON_7 = 25, - GAMEFUNC_WEAPON_8 = 26, - GAMEFUNC_WEAPON_9 = 27, - GAMEFUNC_WEAPON_10 = 28, - GAMEFUNC_INVENTORY = 29, - GAMEFUNC_INVENTORY_LEFT = 30, - GAMEFUNC_INVENTORY_RIGHT = 31, - GAMEFUNC_HOLO_DUKE = 32, - GAMEFUNC_JETPACK = 33, - GAMEFUNC_NIGHTVISION = 34, - GAMEFUNC_MEDKIT = 35, - GAMEFUNC_TURNAROUND = 36, - GAMEFUNC_SENDMESSAGE = 37, - GAMEFUNC_MAP = 38, - GAMEFUNC_SHRINK_SCREEN = 39, - GAMEFUNC_ENLARGE_SCREEN = 40, - GAMEFUNC_CENTER_VIEW = 41, - GAMEFUNC_HOLSTER_WEAPON = 42, - GAMEFUNC_SHOW_OPPONENTS_WEAPON = 43, - GAMEFUNC_MAP_FOLLOW_MODE = 44, - GAMEFUNC_SEE_COOP_VIEW = 45, - GAMEFUNC_MOUSE_AIMING = 46, - GAMEFUNC_TOGGLE_CROSSHAIR = 47, - GAMEFUNC_STEROIDS = 48, - GAMEFUNC_QUICK_KICK = 49, - GAMEFUNC_NEXT_WEAPON = 50, - GAMEFUNC_PREVIOUS_WEAPON = 51, --- GAMEFUNC_SHOW_CONSOLE = 52, - GAMEFUNC_SHOW_DUKEMATCH_SCORES = 53, - GAMEFUNC_DPAD_SELECT = 54, - GAMEFUNC_DPAD_AIMING = 55, -} - -local function shallow_copy(tab) - local t = {} - for k,v in pairs(tab) do - t[k] = v - end - return t -end - --- KEEPINSYNC with gamedef.c:C_AddDefaultDefinitions() and the respective --- defines. These are exported to the ffi.C namespace (except STAT) and as --- literal defines in lunacon.lua. -labels = -{ - STR, - PROJ, - EVENT, - shallow_copy(SFLAG), - setmetatable(STAT, { __metatable="noffiC" }), - GAMEFUNC, -} - -user_sflags = 0 --- TWEAK_SFLAG -for name, flag in pairs(SFLAG) do - if (flag > 0) then - user_sflags = user_sflags + flag - else - SFLAG[name] = -flag - labels[4][name] = nil - end -end - --- KEEPINSYNC player.h -wdata_members = -{ - -- NOTE: they are lowercased for Lunatic - -- NOTE: members _*sound*, _spawn and _shoots assume *zero* to mean "none" - -- (-1 would be more logical). - "const int32_t _workslike", - "int32_t clip", - "int32_t reload", - "int32_t firedelay", - "int32_t totaltime", - "int32_t holddelay", - "int32_t flags", - "const int32_t _shoots", - "int32_t spawntime", - "const int32_t _spawn", - "int32_t shotsperburst", - "const int32_t _initialsound", - "const int32_t _firesound", - "int32_t sound2time", -- NOTE: this is a time number, not a sound - "const int32_t _sound2sound", - "const int32_t _reloadsound1", - "const int32_t _reloadsound2", - "const int32_t _selectsound", - "int32_t flashcolor", -} - - -local SP = function(memb) return "sprite[%s]"..memb end -local ATSP = function(memb) return "_atsprite[%s]"..memb end -local AC = function(memb) return "actor[%s]"..memb end -local SX = function(memb) return "spriteext[%s]"..memb end - --- Generate code to access a signed member as unsigned. -local function s2u(label) - return "(_band("..label.."+65536,65535))" -end - -local function S2U(label) - return { s2u(label), label } -end - --- Some literal checker functions (LITERAL_CHECKING). --- KEEPINSYNC with the actual setter code. -local function litok_gem1(lit) - return (lit >= -1) -end - -local function litok_ge0(lit) - return (lit >= 0) -end - -local ActorLabels = { - x = SP".x", - y = SP".y", - z = SP".z", - cstat = SP".cstat", - picnum = { SP".picnum", SP":set_picnum(%%s)", litok_ge0 }, - shade = SP".shade", - pal = SP".pal", - clipdist = SP".clipdist", --- filler = SP".filler", --- detail = SP".filler", -- NAME - blend = SP".blend", - xrepeat = SP".xrepeat", - yrepeat = SP".yrepeat", - xoffset = SP".xoffset", - yoffset = SP".yoffset", - sectnum = { SP".sectnum", SP":changesect(%%s)", litok_ge0 }, -- set: for tsprite - statnum = { SP".statnum" }, - ang = SP".ang", - owner = { SP".owner", SP":_set_owner(%%s)", litok_ge0 }, - xvel = SP".xvel", - yvel = { SP".yvel", SP":set_yvel(%%s)" }, - zvel = SP".zvel", - lotag = SP".lotag", - hitag = SP".hitag", - extra = SP".extra", - - ulotag = S2U(SP".lotag"), - uhitag = S2U(SP".hitag"), - - -- ActorExtra labels... - htcgg = AC".cgg", - -- XXX: why <0 allowed? - htpicnum = { AC".picnum", AC":set_picnum(%%s)" }, - htang = AC".ang", - htextra = AC".extra", - htowner = { AC".owner", AC":set_owner(%%s)", litok_ge0 }, - htmovflag = AC"._movflag", - httempang = AC".tempang", - htactorstayput = AC".stayputsect", -- NAME - htdispicnum = { AC".dispicnum" }, - -- NOTE: no access for .shootzvel - httimetosleep = AC".timetosleep", - htfloorz = AC".floorz", - htceilingz = AC".ceilingz", - htlastvx = AC".lastvx", - htlastvy = AC".lastvy", - htbposx = AC".bpos.x", - htbposy = AC".bpos.y", - htbposz = AC".bpos.z", - -- Read access differs from write ({ get, set }): - htg_t = { AC":_get_t_data(%s)", AC":_set_t_data(%s,%%s)" }, - htflags = AC".flags", - movflags = AC".movflags", - - -- (mostly) model-related flags - angoff = SX".angoff", - pitch = SX".pitch", - roll = SX".roll", - mdxoff = SX".mdoff.x", -- NAME - mdyoff = SX".mdoff.y", - mdzoff = SX".mdoff.z", - mdflags = SX".flags", - xpanning = SX".xpanning", - ypanning = SX".ypanning", - - alpha = { "_math.floor(spriteext[%s].alpha*255)", "spriteext[%s].alpha=(%%s)/255" }, - - isvalid = { "_con._isvalid(%s)" }, -} - -local function spr2tspr(code) - if (code and code:find(SP"", 1, true)==1) then - return ATSP(code:sub(#SP"" + 1)) - end - -- else return nothing -end - -local TspriteLabels = {} - -for member, code in pairs(ActorLabels) do - if (type(code)=="string") then - TspriteLabels["tspr"..member] = spr2tspr(code) - else - TspriteLabels["tspr"..member] = { spr2tspr(code[1]), spr2tspr(code[2]) } - end -end - --- Sprites set stat- and sectnum via sprite.change{stat,sect} functions. -ActorLabels.sectnum[2] = "sprite.changesect(%s,%%s,true)" -ActorLabels.statnum[2] = "sprite.changestat(%s,%%s,true)" - -local PL = function(memb) return "player[%s]"..memb end --- Access to DukePlayer_t's bool members: they must be read as numbers. -local PLBOOL = function(memb) return { "("..PL(memb).." and 1 or 0)", PL(memb) } end - -local empty_table = {} -local DISABLED_PL = function() return empty_table end -local DISABLED = DISABLED_PL - -local PlayerLabels = { - posx = PL".pos.x", - posy = PL".pos.y", - posz = PL".pos.z", - oposx = PL".opos.x", - oposy = PL".opos.y", - oposz = PL".opos.z", - posxv = PL".vel.x", -- NAME - posyv = PL".vel.y", - poszv = PL".vel.z", - -- NOTE: no access for .npos - bobposx = DISABLED_PL".bobposx", - bobposy = DISABLED_PL".bobposy", - - truefz = DISABLED_PL".truefz", - truecz = DISABLED_PL".truecz", - player_par = PL".player_par", - - randomflamex = DISABLED_PL".randomflamex", - exitx = DISABLED_PL".exitx", - exity = DISABLED_PL".exity", - - runspeed = PL".runspeed", - max_player_health = PL".max_player_health", - max_shield_amount = PL".max_shield_amount", - - autostep = PL".autostep", - autostep_sbw = PL".autostep_sbw", - - interface_toggle_flag = DISABLED_PL".interface_toggle_flag", - - -- NOTE: *bombControl etc. are accessed by gamevars in CON - - max_actors_killed = PL".max_actors_killed", - actors_killed = PL".actors_killed", - - -- NOTE the special case; "%%s" is used to mark settable members - -- with METHOD_MEMBER syntax, it's the value to be set. - gotweapon = { "("..PL":has_weapon(%s) and 1 or 0)", PL":_gt_weapon(%s,%%s)" }, - zoom = PL".zoom", - - loogiex = {}, - loogiey = {}, - - sbs = PL".sbs", - sound_pitch = PL".sound_pitch", - - ang = PL".ang", - oang = PL".oang", - angvel = PL".angvel", - - cursectnum = PL".cursectnum", - - look_ang = PL".look_ang", - last_extra = PL".last_extra", - subweapon = PL".subweapon", - - max_ammo_amount = PL".max_ammo_amount[%s]", - ammo_amount = PL".ammo_amount[%s]", - -- NOTE: no direct access for .inv_amount (but see end) - - wackedbyactor = PL".wackedbyactor", - pyoff = PL".pyoff", - opyoff = PL".opyoff", - - horiz = PL".horiz", - horizoff = PL".horizoff", - ohoriz = PL".ohoriz", - ohorizoff = PL".ohorizoff", - - newowner = PL".newowner", - - jumping_counter = PL".jumping_counter", - airleft = PL".airleft", - - fta = PL".fta", - ftq = PL".ftq", - access_wallnum = { PL".access_wallnum" }, - access_spritenum = { PL".access_spritenum" }, - - got_access = PL".got_access", - weapon_ang = PL".weapon_ang", - visibility = PL".visibility", - - somethingonplayer = PL".somethingonplayer", - on_crane = PL".on_crane", - - i = { PL".i" }, - index = { "%s" }, - - one_parallax_sectnum = DISABLED{ PL".one_parallax_sectnum" }, - - random_club_frame = PL".random_club_frame", - one_eighty_count = PL".one_eighty_count", - - dummyplayersprite = DISABLED_PL".dummyplayersprite", - extra_extra8 = PL".extra_extra8", - - actorsqu = PL".actorsqu", - timebeforeexit = PL".timebeforeexit", - customexitsound = { PL".customexitsound" }, - - last_pissed_time = PL".last_pissed_time", - - weaprecs = PL".weaprecs[%s]", - - weapon_sway = PL".weapon_sway", - crack_time = PL".crack_time", - bobcounter = PL".bobcounter", - - -- NOTE: no access for .orotscrnang - rotscrnang = PL".rotscrnang", - dead_flag = PL".dead_flag", - - holoduke_on = PL".holoduke_on", - pycount = PL".pycount", - transporter_hold = PL".transporter_hold", - - max_secret_rooms = PL".max_secret_rooms", - secret_rooms = PL".secret_rooms", - - frag = PL".frag", - fraggedself = PL".fraggedself", - quick_kick = PL".quick_kick", - last_quick_kick = PL".last_quick_kick", - - return_to_center = DISABLED_PL".return_to_center", - reloading = PLBOOL".reloading", - weapreccnt = { PL".weapreccnt" }, - - aim_mode = PL".aim_mode", - auto_aim = PL".auto_aim", - weaponswitch = PL".weaponswitch", - movement_lock = PL".movement_lock", - team = PL".team", - - tipincs = PL".tipincs", - hbomb_hold_delay = PL".hbomb_hold_delay", - frag_ps = PL".frag_ps", - kickback_pic = PL".kickback_pic", - - gm = PL".gm", - on_warping_sector = PLBOOL".on_warping_sector", - footprintcount = PL".footprintcount", - hurt_delay = PL".hurt_delay", - - hbomb_on = PLBOOL".hbomb_on", - jumping_toggle = PLBOOL".jumping_toggle", - rapid_fire_hold = PLBOOL".rapid_fire_hold", - on_ground = PLBOOL".on_ground", - - inven_icon = PL".inven_icon", - buttonpalette = PL".buttonpalette", - over_shoulder_on = PLBOOL".over_shoulder_on", - show_empty_weapon = PL".show_empty_weapon", - - jetpack_on = PLBOOL".jetpack_on", - spritebridge = PLBOOL".spritebridge", - lastrandomspot = DISABLED_PL".lastrandomspot", - - scuba_on = PLBOOL".scuba_on", - footprintpal = PL".footprintpal", - heat_on = PLBOOL".heat_on", - invdisptime = PL".invdisptime", - holster_weapon = PLBOOL".holster_weapon", - falling_counter = PL".falling_counter", - footprintshade = PL".footprintshade", - - refresh_inventory = PL".refresh_inventory", - last_full_weapon = PL".last_full_weapon", - - walking_snd_toggle = PL".walking_snd_toggle", - palookup = PL".palookup", - hard_landing = PL".hard_landing", - fist_incs = PL".fist_incs", - - toggle_key_flag = { PL".toggle_key_flag" }, - knuckle_incs = PL".knuckle_incs", - knee_incs = PL".knee_incs", - access_incs = PL".access_incs", - - numloogs = DISABLED_PL".numloogs", - loogcnt = PL".loogcnt", - scream_voice = { PL".scream_voice" }, - - last_weapon = PL".last_weapon", - cheat_phase = { PL".cheat_phase" }, - weapon_pos = PL".weapon_pos", - wantweaponfire = PL".wantweaponfire", - - curr_weapon = PL".curr_weapon", - - palette = { PL".palette" }, - - -- NOTE the special case: - pals = PL"._pals[%s]", - pals_time = PL"._pals.f", - - name = {}, - - -- Access to .inv_amount - steroids_amount = PL".inv_amount[0]", - shield_amount = PL".inv_amount[1]", - scuba_amount = PL".inv_amount[2]", - holoduke_amount = PL".inv_amount[3]", - jetpack_amount = PL".inv_amount[4]", - -- 5: dummy - -- 6: no "access_amount" - heat_amount = PL".inv_amount[7]", - -- 8: dummy - firstaid_amount = PL".inv_amount[9]", - boot_amount = PL".inv_amount[10]", -} - -local SEC = function(memb) return "sector[%s]"..memb end -local SECRO = function(memb) return { "sector[%s]"..memb } end - -local SectorLabels = { - wallptr = SECRO".wallptr", - wallnum = SECRO".wallnum", - - ceilingz = SEC".ceilingz", - floorz = SEC".floorz", - - ceilingstat = SEC".ceilingstat", - floorstat = SEC".floorstat", - - -- CEILING - ceilingpicnum = { SEC".ceilingpicnum", SEC":set_ceilingpicnum(%%s)", litok_ge0 }, - - ceilingslope = SEC".ceilingheinum", -- NAME - ceilingshade = SEC".ceilingshade", - - ceilingpal = SEC".ceilingpal", - ceilingxpanning = SEC".ceilingxpanning", - ceilingypanning = SEC".ceilingypanning", - - -- FLOOR - floorpicnum = { SEC".floorpicnum", SEC":set_floorpicnum(%%s)", litok_ge0 }, - - floorslope = SEC".floorheinum", -- NAME - floorshade = SEC".floorshade", - - floorpal = SEC".floorpal", - floorxpanning = SEC".floorxpanning", - floorypanning = SEC".floorypanning", - - visibility = SEC".visibility", - fogpal = SEC".fogpal", - alignto = SEC".fogpal", -- NAME - - lotag = SEC".lotag", - hitag = SEC".hitag", - extra = SEC".extra", - - ceilingbunch = { SEC".ceilingbunch" }, - floorbunch = { SEC".floorbunch" }, - - ulotag = S2U(SEC".lotag"), - uhitag = S2U(SEC".hitag"), -} - -local WAL = function(memb) return "wall[%s]"..memb end -local WALRO = function(memb) return { "wall[%s]"..memb } end - -local WallLabels = { - x = WAL".x", - y = WAL".y", - point2 = WALRO".point2", - nextwall = { WAL".nextwall", WAL":_set_nextwall(%%s)" }, - nextsector = { WAL".nextsector", WAL":_set_nextsector(%%s)" }, - cstat = WAL".cstat", - picnum = { WAL".picnum", WAL":set_picnum(%%s)", litok_ge0 }, - overpicnum = { WAL".overpicnum", WAL":set_overpicnum(%%s)", litok_ge0 }, - shade = WAL".shade", - pal = WAL".pal", - xrepeat = WAL".xrepeat", - yrepeat = WAL".yrepeat", - xpanning = WAL".xpanning", - ypanning = WAL".ypanning", - lotag = WAL".lotag", - hitag = WAL".hitag", - extra = WAL".extra", - blend = WAL".blend", - - ulotag = S2U(WAL".lotag"), - uhitag = S2U(WAL".hitag"), -} - -local function tonegtag(LabelsTab, member, funcname) - local memb = LabelsTab[member] - LabelsTab[member] = { memb, memb.."="..funcname.."(%%s)" } -end - -function setup_negative_tag_check(funcname) - tonegtag(TspriteLabels, "tsprlotag", funcname) - tonegtag(TspriteLabels, "tsprhitag", funcname) - tonegtag(ActorLabels, "lotag", funcname) - tonegtag(ActorLabels, "hitag", funcname) - tonegtag(WallLabels, "lotag", funcname) - tonegtag(WallLabels, "hitag", funcname) - tonegtag(SectorLabels, "lotag", funcname) - tonegtag(SectorLabels, "hitag", funcname) -end - -local PROJ = function(memb) return "projectile[%s]"..memb end -local THISPROJ = function(memb) return "actor[%s].proj"..memb end - -local ProjectileLabels = { - workslike = PROJ".workslike", - cstat = PROJ".cstat", - hitradius = PROJ".hitradius", - range = PROJ".range", - flashcolor = PROJ".flashcolor", - spawns = { PROJ".spawns", PROJ":set_spawns(%%s)", litok_gem1 }, - sound = { PROJ".sound", PROJ":set_sound(%%s)", litok_gem1 }, - isound = { PROJ".isound", PROJ":set_isound(%%s)", litok_gem1 }, - vel = PROJ".vel", - decal = { PROJ".decal", PROJ":set_decal(%%s)", litok_gem1 }, - trail = { PROJ".trail", PROJ":set_trail(%%s)", litok_gem1 }, - tnum = PROJ".tnum", - drop = PROJ".drop", - offset = PROJ".offset", - bounces = PROJ".bounces", - bsound = { PROJ".bsound", PROJ":set_bsound(%%s)", litok_gem1 }, - toffset = PROJ".toffset", - extra = PROJ".extra", - extra_rand = PROJ".extra_rand", - sxrepeat = PROJ".sxrepeat", - syrepeat = PROJ".syrepeat", - txrepeat = PROJ".txrepeat", - tyrepeat = PROJ".tyrepeat", - shade = PROJ".shade", - xrepeat = PROJ".xrepeat", - yrepeat = PROJ".yrepeat", - pal = PROJ".pal", - velmult = PROJ".movecnt", -- NAME - clipdist = PROJ".clipdist", - userdata = PROJ".userdata", -} - --- XXX: kind of CODEDUP form spr2tspr -local function proj2thisproj(code) - if (code and code:find(PROJ"", 1, true)==1) then - return THISPROJ(code:sub(#PROJ"" + 1)) - end - -- else return nothing -end - -local SpriteProjectileLabels = {} - -for member, code in pairs(ProjectileLabels) do - if (type(code)=="string") then - SpriteProjectileLabels[member] = proj2thisproj(code) - else - SpriteProjectileLabels[member] = { proj2thisproj(code[1]), proj2thisproj(code[2]) } - end -end - -local UD = function(memb) return "_gud(_pli)"..memb end -local UDRO = function(memb) return { UD(memb) } end - --- NOTE: Only members that are actually encountered in existing mods are added here. --- TODO: r5043, r5044 -local UserdefLabels = { - althud = UD".althud", - auto_run = UD".auto_run", - camerasprite = UDRO".camerasprite", - cashman = UDRO".cashman", - clipping = UD".noclip", -- NAME - color = UD".color", - const_visibility = UD".const_visibility", - crosshair = UD".crosshair", - crosshairscale = UDRO".crosshairscale", - detail = { "1" }, - display_bonus_screen = UD".display_bonus_screen", - drawweapon = UDRO".drawweapon", - eog = UD".eog", - ffire = UDRO".ffire", - fta_on = UD".fta_on", - god = UD".god", - idplayers = UDRO".idplayers", - last_level = UDRO".last_level", - level_number = { UD".level_number", UD":set_level_number(%%s)", {0, MAXLEVELS-1} }, - levelstats = UD".levelstats", - lockout = UDRO".lockout", - m_origin_x = UD".m_origin.x", - m_origin_y = UD".m_origin.y", - m_player_skill = UDRO".m_player_skill", - m_volume_number = { UD".m_volume_number", UD":set_m_volume_number(%%s)", {0, MAXVOLUMES} }, - mouseaiming = UD".mouseaiming", - pause_on = UDRO".pause_on", - player_skill = UD".player_skill", - playerbest = UDRO".playerbest", - mouseflip = UDRO".mouseflip", - multimode = { "1" }, - musictoggle = UDRO".config.MusicToggle", - noexits = UDRO".noexits", - overhead_on = UD".overhead_on", - recstat = UDRO".recstat", - runkey_mode = UD".runkey_mode", - show_level_text = UD".show_level_text", - screen_size = { UD".screen_size", UD":set_screen_size(%%s)" }, - screen_tilting = UD".screen_tilting", - showallmap = UD".showallmap", - showweapons = UDRO".showweapons", - statusbarmode = UDRO".statusbarmode", - statusbarscale = UDRO".statusbarscale", - volume_number = { UD".volume_number", UD":set_volume_number(%%s)", {0, MAXVOLUMES} }, - weaponscale = UDRO".weaponscale", - weaponswitch = UD".weaponswitch", -} - -local INP = function(memb) return PL"._input"..memb end - -local InputLabels = { - avel = INP".avel", - horz = INP".horz", - fvel = INP".fvel", - svel = INP".svel", - bits = INP".bits", - extbits = INP".extbits", -} - -local TileDataLabels = { - -- tilesiz[] - xsize = "g_tile.sizx[%s]", - ysize = "g_tile.sizy[%s]", - - -- picanm[] --- "animframes", --- "xoffset", --- "yoffset", --- "animspeed", --- "animtype", - - -- g_tile[] - gameflags = { "g_tile[%s]._flags" }, -} - -StructAccessCode = -{ - sector = SectorLabels, - wall = WallLabels, - sprite = ActorLabels, - player = PlayerLabels, - tspr = TspriteLabels, - projectile = ProjectileLabels, - thisprojectile = SpriteProjectileLabels, - userdef = UserdefLabels, - input = InputLabels, - tiledata = TileDataLabels, --- TODO: tiledata picanm[] members, paldata -} - --- NOTE: These MUST be in reverse lexicographical order! --- Per CON syntax, valid identifiers names are disjoint from keywords, --- so that a rule like --- t_identifier = -con_keyword * (sp1 + "[") * t_identifier_all --- (from the final grammar in lunacon.lua) must match the longest --- possible keyword name, else the negation might wrongly not fail. - -keyword = - -lpeg.P(false) + -"}" + -"{" + -"zshootvar" + -"zshoot" + -"xorvarvar" + -"xorvar" + -"writearraytofile" + -"whilevarvarn" + -"whilevarn" + -"wackplayer" + -"userquote" + -"useractor" + -"updatesectorz" + -"updatesector" + -"undefinevolume" + -"undefineskill" + -"undefinelevel" + -"tossweapon" + -"tip" + -"time" + -"switch" + -"subvarvar" + -"subvar" + -"strength" + -"stopsoundvar" + -"stopsound" + -"stopallsounds" + -"stopactorsound" + -"state" + -"starttrackvar" + -"starttrack" + -"startlevel" + -"startcutscene" + -"ssp" + -"sqrt" + -"spriteshadow" + -"spritepal" + -"spritenvg" + -"spritenoshade" + -"spritenopal" + -"spriteflags" + -"spgetlotag" + -"spgethitag" + -"spawn" + -"soundvar" + -"soundoncevar" + -"soundonce" + -"sound" + -"smaxammo" + -"sleeptime" + -"sizeto" + -"sizeat" + -"sin" + -"showviewunbiased" + -"showview" + -"shootvar" + -"shoot" + -"shiftvarr" + -"shiftvarl" + -"shadeto" + -"setwall" + -"setvarvar" + -"setvar" + -"setuserdef" + -"settspr" + -"setthisprojectile" + -"setsprite" + -"setsector" + -"setprojectile" + -"setplayervar" + -"setplayerangle" + -"setplayer" + -"setmusicposition" + -"setinput" + -"setgamepalette" + -"setgamename" + -"setdefname" + -"setcfgname" + -"setaspect" + -"setarray" + -"setactorvar" + -"setactorsoundpitch" + -"setactorangle" + -"setactor" + -"sectsetinterpolation" + -"sectorofwall" + -"sectgetlotag" + -"sectgethitag" + -"sectclearinterpolation" + -"scriptsize" + -"screentext" + -"screensound" + -"savenn" + -"savemapstate" + -"savegamevar" + -"save" + -"rotatespritea" + -"rotatesprite16" + -"rotatesprite" + -"rotatepoint" + -"return" + -"respawnhitag" + -"resizearray" + -"resetplayerflags" + -"resetplayer" + -"resetcount" + -"resetactioncount" + -"redefinequote" + -"readgamevar" + -"readarrayfromfile" + -"rayintersect" + -"randvarvar" + -"randvar" + -"quote" + -"quake" + -"qsubstr" + -"qstrncat" + -"qstrlen" + -"qstrdim" + -"qstrcpy" + -"qstrcat" + -"qsprintf" + -"qspawnvar" + -"qspawn" + -"qgetsysstr" + -"pstomp" + -"prevspritestat" + -"prevspritesect" + -"precache" + -"pkick" + -"paper" + -"palfrom" + -"orvarvar" + -"orvar" + -"operatesectors" + -"operaterespawns" + -"operatemasterswitches" + -"operateactivators" + -"operate" + -"onevent" + -"nullop" + -"nextspritestat" + -"nextspritesect" + -"neartag" + -"myosx" + -"myospalx" + -"myospal" + -"myos" + -"music" + -"mulvarvar" + -"mulvar" + -"mulscale" + -"movesprite" + -"move" + -"money" + -"modvarvar" + -"modvar" + -"minitext" + -"mikesnd" + -"mail" + -"lotsofglass" + -"lockplayer" + -"loadmapstate" + -"lineintersect" + -"ldist" + -"killit" + -"jump" + -"insertspriteq" + -"inittimer" + -"includedefault" + -"include" + -"ifwasweapon" + -"ifvarxor" + -"ifvarvarxor" + -"ifvarvaror" + -"ifvarvarn" + -"ifvarvarl" + -"ifvarvarg" + -"ifvarvareither" + -"ifvarvare" + -"ifvarvarand" + -"ifvaror" + -"ifvarn" + -"ifvarl" + -"ifvarg" + -"ifvareither" + -"ifvare" + -"ifvarand" + -"ifstrength" + -"ifsquished" + -"ifspritepal" + -"ifspawnedby" + -"ifsound" + -"ifserver" + -"ifrnd" + -"ifrespawn" + -"ifplayersl" + -"ifpinventory" + -"ifphealthl" + -"ifpdistl" + -"ifpdistg" + -"ifp" + -"ifoutside" + -"ifonwater" + -"ifnotmoving" + -"ifnosounds" + -"ifmultiplayer" + -"ifmove" + -"ifinwater" + -"ifinspace" + -"ifinouterspace" + -"ifhitweapon" + -"ifhitspace" + -"ifgotweaponce" + -"ifgapzl" + -"iffloordistl" + -"ifdead" + -"ifcutscene" + -"ifcount" + -"ifclient" + -"ifceilingdistl" + -"ifcanshoottarget" + -"ifcanseetarget" + -"ifcansee" + -"ifbulletnear" + -"ifawayfromwall" + -"ifangdiffl" + -"ifai" + -"ifactorsound" + -"ifactornotstayput" + -"ifactor" + -"ifactioncount" + -"ifaction" + -"hitscan" + -"hitradiusvar" + -"hitradius" + -"headspritestat" + -"headspritesect" + -"guts" + -"guniqhudid" + -"gmaxammo" + -"globalsoundvar" + -"globalsound" + -"getzrange" + -"getwall" + -"getuserdef" + -"gettspr" + -"gettimedate" + -"getticks" + -"getthisprojectile" + -"gettexturefloor" + -"gettextureceiling" + -"getsector" + -"getprojectile" + -"getpname" + -"getplayervar" + -"getplayerangle" + -"getplayer" + -"getmusicposition" + -"getlastpal" + -"getkeyname" + -"getinput" + -"getincangle" + -"getflorzofslope" + -"getcurraddress" + -"getceilzofslope" + -"getarraysize" + -"getangletotarget" + -"getangle" + -"getactorvar" + -"getactorangle" + -"getactor" + -"gamevar" + -"gametextz" + -"gametext" + -"gamestartup" + -"gamearray" + -"flash" + -"findplayer" + -"findotherplayer" + -"findnearspritezvar" + -"findnearspritez" + -"findnearspritevar" + -"findnearsprite3dvar" + -"findnearsprite3d" + -"findnearsprite" + -"findnearactorzvar" + -"findnearactorz" + -"findnearactorvar" + -"findnearactor3dvar" + -"findnearactor3d" + -"findnearactor" + -"fall" + -"ezshootvar" + -"ezshoot" + -"eventloadactor" + -"espawnvar" + -"espawn" + -"eshootvar" + -"eshoot" + -"eqspawnvar" + -"eqspawn" + -"enhanced" + -"endswitch" + -"ends" + -"endoflevel" + -"endofgame" + -"endevent" + -"enda" + -"else" + -"echo" + -"dynamicsoundremap" + -"dynamicremap" + -"dragpoint" + -"divvarvar" + -"divvar" + -"dist" + -"displayrandvarvar" + -"displayrandvar" + -"displayrand" + -"digitalnumberz" + -"digitalnumber" + -"defstate" + -"definevolumename" + -"definevolumeflags" + -"definesound" + -"defineskillname" + -"definequote" + -"defineprojectile" + -"definelevelname" + -"definegametype" + -"definegamefuncname" + -"definecheat" + -"define" + -"default" + -"debug" + -"debris" + -"cstator" + -"cstat" + -"count" + -"cos" + -"copy" + -"cmenu" + -"clipmovenoslide" + -"clipmove" + -"clipdist" + -"clearmapstate" + -"checkavailweapon" + -"checkavailinven" + -"checkactivatormotion" + -"cheatkeys" + -"changespritestat" + -"changespritesect" + -"case" + -"canseespr" + -"cansee" + -"calchypotenuse" + -"cactor" + -"break" + -"betaname" + -"appendevent" + -"angoffvar" + -"angoff" + -"andvarvar" + -"andvar" + -"ai" + -"addweaponvar" + -"addweapon" + -"addvarvar" + -"addvar" + -"addstrength" + -"addphealth" + -"addlogvar" + -"addlog" + -"addkills" + -"addinventory" + -"addammo" + -"actor" + -"activatecheat" + -"activatebysector" + -"activate" + -"action" + -lpeg.P(false) diff --git a/source/duke3d/src/lunatic/control.lua b/source/duke3d/src/lunatic/control.lua deleted file mode 100644 index 550c44767..000000000 --- a/source/duke3d/src/lunatic/control.lua +++ /dev/null @@ -1,2420 +0,0 @@ --- Game control module for Lunatic. - -local require = require -local ffi = require("ffi") -local ffiC = ffi.C -local jit = require("jit") - --- Lua C API functions, this comes from El_PushCFunctions() in lunatic_game.c. -local CF = CF - -local bit = require("bit") -local debug = require("debug") -local io = require("io") -local math = require("math") -local table = require("table") - -local bcheck = require("bcheck") -local con_lang = require("con_lang") - -local byte = require("string").byte -local setmetatable = setmetatable - -local band, bor = bit.band, bit.bor -local rshift = bit.rshift -local tobit = bit.tobit - -local floor = math.floor - -local assert = assert -local error = error -local ipairs = ipairs -local pairs = pairs -local print = print -local rawget = rawget -local rawset = rawset -local select = select -local tostring = tostring -local type = type -local unpack = unpack - -local format = require("string").format - -local actor, player = assert(actor), assert(player) -local dc = require("defs_common") -local cansee, hitscan, neartag = dc.cansee, dc.hitscan, dc.neartag -local inside = dc.inside - -local sector, wall, sprite = dc.sector, dc.wall, dc.sprite -local wallsofsect = dc.wallsofsect -local spritesofsect, spritesofstat = dc.spritesofsect, dc.spritesofstat - -local check_sector_idx = bcheck.sector_idx -local check_tile_idx = bcheck.tile_idx -local check_sprite_idx = bcheck.sprite_idx -local check_player_idx = bcheck.player_idx -local check_sound_idx = bcheck.sound_idx -local check_number = bcheck.number -local check_type = bcheck.type - -local lprivate = require("lprivate") -local GET, WEAPON = lprivate.GET, lprivate.WEAPON - -ffi.cdef[[ -size_t fwrite(const void * restrict ptr, size_t size, size_t nmemb, void * restrict stream); -]] - -local OUR_REQUIRE_STRING = [[ - local _con=require'con' - local _ga,_av,_pv=_con._gamearray,_con.actorvar,_con.playervar -]] -local function our_get_require() - return OUR_REQUIRE_STRING -end - - -module(...) - - ----=== ACTION/MOVE/AI HELPERS ===--- - -local lastid = { action=0, move=0, ai=0 } - -local con_action_ct = ffi.typeof("const con_action_t") -local con_move_ct = ffi.typeof("const con_move_t") -local con_ai_ct = ffi.typeof("const con_ai_t") - --- All-zero action and move with IDs. Mostly for CON support. -local literal_act = { [0]=con_action_ct(0), [1]=con_action_ct(1) } -local literal_mov = { [0]=con_move_ct(0), [1]=con_move_ct(1) } - -local literal_am = { action=literal_act, move=literal_mov } --- Const-qualified 'full' action and move (with ID): -local am_ctype_full_const = { action=con_action_ct, move=con_move_ct } --- Non-const-qualified 'bare' action and move (without ID): -local am_ctype_bare = { action=ffi.typeof("struct action"), move=ffi.typeof("struct move") } - --- CODEDUP lunacon.lua -local function truetab(tab) - local ttab = {} - for i=1,#tab do - ttab[tab[i]] = true - end - return ttab -end - --- KEEPINSYNC lunacon.lua -local ALLOWED_VIEWTYPE = truetab { 0, 1, 2, 3,4, 5, 7, 8, -5, -7, -8 } - -local function def_action_or_move(what, tab) - if (lastid[what] <= -(2^31)) then - error("Too many "..what.."s defined", 3); - end - - bcheck.top_level(what, 4) - - -- NOTE: tab[0]~=nil check for "Special default values" below. - if (type(tab) ~= "table" or tab[0]~=nil) then - error("invalid argument to con."..what..": must be a table", 3) - end - - -- Pass args table to ffi.new, which can take either: a table with numeric - -- indices, or a table with key-value pairs, *but not in combination*. - -- See http://luajit.org/ext_ffi_semantics.html#init_table - local am = am_ctype_bare[what](tab) - - -- Now, set all string keys as they have been ignored if tab[1] was - -- non-nil. - for key, val in pairs(tab) do - if (type(key)=="string") then - am[key] = val - end - end - - if (what=="action") then - -- Special default values or checking of actor members. - -- KEEPINSYNC with ACTOR_CHECK in lunacon.lua for consistency. - local numframes = tab[2] or tab.numframes - local viewtype = tab[3] or tab.viewtype - local incval = tab[4] or tab.incval - - if (numframes==nil) then - am.numframes = 1 - else - check_number(numframes, 4) - if (numframes < 0) then - error("action has negative number of frames", 3) - end - end - - if (viewtype==nil) then - am.viewtype = 1 - else - check_number(viewtype, 4) - if (ALLOWED_VIEWTYPE[viewtype] == nil) then - error("action has disallowed viewtype "..viewtype, 3) - end - end - - if (incval==nil) then - am.incval = 1 - end - end - - -- Named actions or moves have negative ids so that non-negative ones - -- can be used as (different) placeholders for all-zero ones. - lastid[what] = lastid[what]-1 - - return am_ctype_full_const[what](lastid[what], am) -end - ----=== ACTION/MOVE/AI FUNCTIONS ===--- - -function action(tab) - return def_action_or_move("action", tab) -end - -function move(tab) - return def_action_or_move("move", tab) -end - --- Get action or move for an 'ai' definition. -local function get_action_or_move(what, val, argi) - if (val == nil) then - return literal_am[what][0] - elseif (ffi.istype(am_ctype_full_const[what], val)) then - return val - elseif (type(val)=="number") then - if (val==0 or val==1) then - return literal_am[what][val] - end - end - - error("bad argument #"..argi.." to ai: must be nil/nothing, 0, 1, or "..what, 3) -end - -function ai(action, move, flags) - bcheck.top_level("ai") - - if (lastid.ai <= -(2^31)) then - error("Too many AIs defined", 2); - end - - local act = get_action_or_move("action", action, 2) - local mov = get_action_or_move("move", move, 3) - - if (flags~=nil) then - if (type(flags)~="number" or not (flags>=0 and flags<=32767)) then - error("bad argument #4 to ai: must be a number in [0..32767]", 2) - end - else - flags = 0 - end - - lastid.ai = lastid.ai-1 - return con_ai_ct(lastid.ai, act, mov, flags) -end - - ----=== RUNTIME CON FUNCTIONS ===--- - --- Will contain [