X-Git-Url: https://deadsoftware.ru/gitweb?a=blobdiff_plain;f=src%2Fgame%2Fg_main.pas;h=def9f2914d20b342539c8021c605dc93e39c353f;hb=9b4492224e2e59560931496e925333bdbe6f2b0e;hp=9a4e345d0206945099bdecbdbdb02275177283d9;hpb=6356457c45922035e2040452cef266c2fc628ece;p=d2df-sdl.git diff --git a/src/game/g_main.pas b/src/game/g_main.pas index 9a4e345..def9f29 100644 --- a/src/game/g_main.pas +++ b/src/game/g_main.pas @@ -28,10 +28,6 @@ procedure KeyPress (K: Word); procedure CharPress (C: AnsiChar); var - {--- TO REMOVE ---} - //GameDir: string; - {-----------------} - {--- Read-only dirs ---} GameWAD: string; DataDirs: SSArray; @@ -60,6 +56,12 @@ uses {$ENDIF} {$IFDEF LINUX} BaseUnix, +{$ENDIF} +{$IFDEF DARWIN} + MacOSAll, CocoaAll, +{$ENDIF} +{$IFDEF USE_SDL2} + SDL2, {$ENDIF} wadreader, e_log, g_window, e_graphics, e_input, g_game, g_console, g_gui, @@ -73,8 +75,7 @@ uses var charbuff: packed array [0..15] of AnsiChar; binPath: AnsiString = ''; - forceCurrentDir: Boolean = false; - + forceBinDir: Boolean; function GetBinaryPath (): AnsiString; {$IFDEF LINUX} @@ -98,140 +99,305 @@ begin if (length(result) > 0) and (result[length(result)] <> '/') then result := result+'/'; end; +procedure PrintDirs (msg: AnsiString; dirs: SSArray); + var dir: AnsiString; +begin + e_LogWriteln(msg + ':'); + for dir in dirs do + e_LogWriteln(' ' + dir); +end; + +{$IFDEF DARWIN} + function NSStringToAnsiString (s: NSString): AnsiString; + var i: Integer; + begin + result := ''; + for i := 0 to s.length - 1 do + result := result + AnsiChar(s.characterAtIndex(i)); + end; + + function GetBundlePath (): AnsiString; + var pathRef: CFURLRef; pathCFStr: CFStringRef; pathStr: ShortString; + begin + pathRef := CFBundleCopyBundleURL(CFBundleGetMainBundle()); + pathCFStr := CFURLCopyFileSystemPath(pathRef, kCFURLPOSIXPathStyle); + CFStringGetPascalString(pathCFStr, @pathStr, 255, CFStringGetSystemEncoding()); + CFRelease(pathRef); + CFRelease(pathCFStr); + Result := pathStr; + end; +{$ENDIF} procedure InitPath; - var i: Integer; rwdir, rodir: AnsiString; - //first: Boolean = true; + var i: Integer; rwdir, rodir: AnsiString; rwdirs, rodirs: SSArray; - procedure xput (s: AnsiString); - { - var - f: TextFile; + procedure AddDir (var dirs: SSArray; append: AnsiString); begin - AssignFile(f, 'zzz.log'); - if (first) then - begin - Rewrite(f); - first := false; - end - else - begin - Append(f); - end; - writeln(f, s); - CloseFile(f); + SetLength(dirs, Length(dirs) + 1); + dirs[High(dirs)] := ExpandFileName(append) end; - } + + function IsSep (ch: Char): Boolean; begin + {$IFDEF WINDOWS} + result := (ch = '/') or (ch = '\'); + {$ELSE} + result := (ch = '/'); + {$ENDIF} end; - procedure AddPath (var arr: SSArray; str: AnsiString); - var - ss: ShortString; + function OptimizePath (dir: AnsiString): AnsiString; + var i, len: Integer; s: AnsiString; begin - if (length(str) = 0) then exit; - if (forceCurrentDir) then + i := 1; len := Length(dir); s := ''; + while i <= len do begin - str := fixSlashes(ExpandFileName(str)); - end - else - begin - str := fixSlashes(str); - if (str[1] <> '/') then str := binPath+str; - while (length(str) > 0) do + if IsSep(dir[i]) then begin - if (str = '/') then exit; - if (str[length(str)] = '/') then begin Delete(str, length(str), 1); continue; end; - if (length(str) >= 2) and (Copy(str, length(str)-1, 2) = '/.') then begin Delete(str, length(str)-1, 2); continue; end; - break; - end; + s := s + DirectorySeparator; + Inc(i); + while (i <= len) and IsSep(dir[i]) do Inc(i); + if (i <= len) and (dir[i] = '.') then + begin + if (i = len) or IsSep(dir[i + 1]) then + begin + Inc(i) + end + else if (i + 1 <= len) and (dir[i + 1] = '.') then + begin + if (i + 1 = len) or IsSep(dir[i + 2]) then + begin + s := e_UpperDir(s); + Inc(i, 2) + end + end + end + end + else + begin + s := s + dir[i]; + Inc(i) + end end; - if (length(str) = 0) then exit; - if (length(str) > 255) then + result := s + end; + + procedure OptimizeDirs (var dirs: SSArray); + var i, j, k: Integer; + begin + for i := 0 to High(dirs) do + dirs[i] := OptimizePath(dirs[i]); + // deduplicate + i := High(dirs); + while i >= 0 do begin - xput('path too long: ['+str+']'); - raise Exception.Create(Format('path "%s" too long', [str])); - end; - for ss in arr do + j := 0; + while j < i do + begin + if dirs[j] = dirs[i] then + begin + for k := j + 1 to High(dirs) do + dirs[k - 1] := dirs[k]; + Dec(i); + SetLength(dirs, High(dirs)) + end + else + begin + Inc(j) + end + end; + Dec(i) + end + end; + + procedure AddDef (var dirs: SSArray; base: SSArray; append: AnsiString); + var s: AnsiString; + begin + if Length(dirs) = 0 then + for s in base do + AddDir(dirs, e_CatPath(s, append)); + OptimizeDirs(dirs) + end; + + function GetDefaultRODirs (): SSArray; + {$IF DEFINED(UNIX) AND NOT DEFINED(DARWIN) AND NOT DEFINED(ANDROID)} + var home: AnsiString; + {$ENDIF} + {$IFDEF WINDOWS} + var appdata: AnsiString; + {$ENDIF} + {$IFDEF DARWIN} + var bundle, s: AnsiString; dirArr: NSArray; i: Integer; + {$ENDIF} + begin + result := nil; + {$IFDEF DARWIN} + bundle := GetBundlePath(); + if ExtractFileExt(bundle) <> '.app' then + AddDir(result, binpath); + {$ELSE} + AddDir(result, binPath); + {$ENDIF} + if forceBinDir = false then begin - //writeln('<<<', ss, '>>> : [', str, ']'); - if (ss = str) then exit; - end; - SetLength(arr, Length(arr)+1); - //arr[High(arr)] := ExpandFileName(str); - arr[High(arr)] := str; - xput('NEW PATH: ['+str+']'); + {$IFDEF USE_SDL2} + AddDir(result, SDL_GetBasePath()); + AddDir(result, SDL_GetPrefPath('', 'doom2df')); + {$ENDIF} + {$IFDEF WINDOWS} + appdata := GetEnvironmentVariable('APPDATA') + '\doom2df'; + if appdata <> '' then + AddDir(result, appdata); + {$ENDIF} + {$IF DEFINED(UNIX) AND NOT DEFINED(DARWIN) AND NOT DEFINED(ANDROID)} + AddDir(result, '/usr/share/doom2df'); + AddDir(result, '/usr/local/share/doom2df'); + home := GetEnvironmentVariable('HOME'); + if home <> '' then + AddDir(result, e_CatPath(home, '.doom2df')); + {$ENDIF} + {$IFDEF DARWIN} + bundle := GetBundlePath(); + if bundle <> '' then + AddDir(result, e_CatPath(bundle, 'Contents/Resources')); + dirArr := NSSearchPathForDirectoriesInDomains(NSApplicationSupportDirectory, NSUserDomainMask, true); + for i := 0 to dirArr.count - 1 do + begin + s := NSStringToAnsiString(dirArr.objectAtIndex(i)); + AddDir(result, e_CatPath(s, 'Doom 2D Forever')) + end; + {$ENDIF} + {$IF DEFINED(ANDROID) AND DEFINED(USE_SDL2)} + AddDir(result, SDL_AndroidGetInternalStoragePath()); + if SDL_AndroidGetExternalStorageState() <> 0 then + AddDir(result, SDL_AndroidGetExternalStoragePath()); + {$ENDIF} + end end; - procedure AddDef (var arr: SSArray; str: AnsiString); - begin - if (length(arr) = 0) then AddPath(arr, str) + function GetDefaultRWDirs (): SSArray; + {$IF DEFINED(UNIX) AND NOT DEFINED(DARWIN) AND NOT DEFINED(ANDROID)} + var home: AnsiString; + {$ENDIF} + {$IFDEF WINDOWS} + var appdata: AnsiString; + {$ENDIF} + {$IFDEF DARWIN} + var bundle, s: AnsiString; dirArr: NSArray; i: Integer; + {$ENDIF} + begin + result := nil; + {$IFDEF DARWIN} + bundle := GetBundlePath(); + if ExtractFileExt(bundle) <> '.app' then + AddDir(result, binPath); + {$ELSE} + AddDir(result, binPath); + {$ENDIF} + if forceBinDir = false then + begin + {$IFDEF USE_SDL2} + AddDir(result, SDL_GetPrefPath('', 'doom2df')); + {$ENDIF} + {$IFDEF WINDOWS} + appdata := GetEnvironmentVariable('APPDATA') + '\doom2df'; + if appdata <> '' then + AddDir(result, appdata); + {$ENDIF} + {$IF DEFINED(UNIX) AND NOT DEFINED(DARWIN) AND NOT DEFINED(ANDROID)} + home := GetEnvironmentVariable('HOME'); + if home <> '' then + AddDir(result, e_CatPath(home, '.doom2df')); + {$ENDIF} + {$IFDEF DARWIN} + dirArr := NSSearchPathForDirectoriesInDomains(NSApplicationSupportDirectory, NSUserDomainMask, true); + for i := 0 to dirArr.count - 1 do + begin + s := NSStringToAnsiString(dirArr.objectAtIndex(i)); + AddDir(result, e_CatPath(s, 'Doom 2D Forever')) + end; + {$ENDIF} + {$IF DEFINED(ANDROID) AND DEFINED(USE_SDL2)} + if SDL_AndroidGetExternalStorageState() <> 0 then + AddDir(result, SDL_AndroidGetExternalStoragePath()); + {$ENDIF} + end end; begin - //GetDir(0, GameDir); + forceBinDir := false; binPath := GetBinaryPath(); - xput('binPath=['+binPath+']'); - - for i := 1 to ParamCount do if (ParamStr(i) = '--cwd') then begin forceCurrentDir := true; break; end; i := 1; while i < ParamCount do begin case ParamStr(i) of + '--like-windoze': forceBinDir := true; '--rw-dir': begin Inc(i); rwdir := ParamStr(i); (* RW *) - AddPath(LogDirs, e_CatPath(rwdir, '')); - AddPath(SaveDirs, e_CatPath(rwdir, 'data')); - AddPath(CacheDirs, e_CatPath(rwdir, 'data/cache')); - AddPath(ConfigDirs, e_CatPath(rwdir, '')); - AddPath(MapDownloadDirs, e_CatPath(rwdir, 'maps/downloads')); - AddPath(WadDownloadDirs, e_CatPath(rwdir, 'wads/downloads')); - AddPath(ScreenshotDirs, e_CatPath(rwdir, 'screenshots')); + AddDir(LogDirs, e_CatPath(rwdir, '')); + AddDir(SaveDirs, e_CatPath(rwdir, 'data')); + AddDir(CacheDirs, e_CatPath(rwdir, 'data/cache')); + AddDir(ConfigDirs, e_CatPath(rwdir, '')); + AddDir(MapDownloadDirs, e_CatPath(rwdir, 'maps/downloads')); + AddDir(WadDownloadDirs, e_CatPath(rwdir, 'wads/downloads')); + AddDir(ScreenshotDirs, e_CatPath(rwdir, 'screenshots')); (* RO *) - AddPath(DataDirs, e_CatPath(rwdir, 'data')); - AddPath(ModelDirs, e_CatPath(rwdir, 'data/models')); - AddPath(MegawadDirs, e_CatPath(rwdir, 'maps/megawads')); - AddPath(MapDirs, e_CatPath(rwdir, 'maps')); - AddPath(WadDirs, e_CatPath(rwdir, 'wads')); + AddDir(DataDirs, e_CatPath(rwdir, 'data')); + AddDir(ModelDirs, e_CatPath(rwdir, 'data/models')); + AddDir(MegawadDirs, e_CatPath(rwdir, 'maps/megawads')); + AddDir(MapDirs, e_CatPath(rwdir, 'maps')); + AddDir(WadDirs, e_CatPath(rwdir, 'wads')); end; '--ro-dir': begin Inc(i); rodir := ParamStr(i); (* RO *) - AddPath(DataDirs, e_CatPath(rodir, 'data')); - AddPath(ModelDirs, e_CatPath(rodir, 'data/models')); - AddPath(MegawadDirs, e_CatPath(rodir, 'maps/megawads')); - AddPath(MapDirs, e_CatPath(rodir, 'maps')); - AddPath(WadDirs, e_CatPath(rodir, 'wads')); + AddDir(DataDirs, e_CatPath(rodir, 'data')); + AddDir(ModelDirs, e_CatPath(rodir, 'data/models')); + AddDir(MegawadDirs, e_CatPath(rodir, 'maps/megawads')); + AddDir(MapDirs, e_CatPath(rodir, 'maps')); + AddDir(WadDirs, e_CatPath(rodir, 'wads')); end; end; Inc(i) end; + // prefer bin dir if it writable and contains game.wad + if forceBinDir = false then + begin + if findDiskWad(binPath + 'data' + '/' + 'GAME') <> '' then + if e_CanCreateFilesAt(binPath) then + forceBinDir := true + end; + (* RO *) - AddDef(DataDirs, 'data'); - AddDef(ModelDirs, 'data/models'); - AddDef(MegawadDirs, 'maps/megawads'); - AddDef(MapDirs, 'maps'); - AddDef(WadDirs, 'wads'); + rodirs := GetDefaultRODirs(); + AddDef(DataDirs, rodirs, 'data'); + AddDef(ModelDirs, rodirs, 'data/models'); + AddDef(MegawadDirs, rodirs, 'maps/megawads'); + AddDef(MapDirs, rodirs, 'maps'); + AddDef(WadDirs, rodirs, 'wads'); + (* RW *) - AddDef(LogDirs, '.'); - AddDef(SaveDirs, 'data'); - AddDef(CacheDirs, 'data/cache'); - AddDef(ConfigDirs, '.'); - AddDef(MapDownloadDirs, 'maps/downloads'); - AddDef(WadDownloadDirs, 'wads/downloads'); - AddDef(ScreenshotDirs, 'screenshots'); + rwdirs := GetDefaultRWDirs(); + AddDef(LogDirs, rwdirs, ''); + AddDef(SaveDirs, rwdirs, 'data'); + AddDef(CacheDirs, rwdirs, 'data/cache'); + AddDef(ConfigDirs, rwdirs, ''); + AddDef(MapDownloadDirs, rwdirs, 'maps/downloads'); + AddDef(WadDownloadDirs, rwdirs, 'wads/downloads'); + AddDef(ScreenshotDirs, rwdirs, 'screenshots'); for i := 0 to High(MapDirs) do - AddPath(AllMapDirs, MapDirs[i]); + AddDir(AllMapDirs, MapDirs[i]); for i := 0 to High(MegawadDirs) do - AddPath(AllMapDirs, MegawadDirs[i]); + AddDir(AllMapDirs, MegawadDirs[i]); + OptimizeDirs(AllMapDirs); if LogFileName = '' then begin @@ -241,65 +407,98 @@ begin {$IFDEF HEADLESS} LogFileName := e_CatPath(rwdir, 'Doom2DF_H.log'); {$ELSE} - LogFileName := e_Catpath(rwdir, 'Doom2DF.log'); + LogFileName := e_CatPath(rwdir, 'Doom2DF.log'); {$ENDIF} end end; - - xput('binPath=['+binPath+']'); end; -procedure Main(); -{$IFDEF ENABLE_HOLMES} - var flexloaded: Boolean; -{$ENDIF} - var s: AnsiString; +procedure InitPrep; + {$IF DEFINED(ANDROID) AND DEFINED(USE_SDLMIXER)} + var timiditycfg: AnsiString; + {$ENDIF} + var i: Integer; begin - InitPath; + {$IFDEF HEADLESS} + conbufDumpToStdOut := true; + {$ENDIF} + for i := 1 to ParamCount do + begin + if (ParamStr(i) = '--con-stdout') then + begin + conbufDumpToStdOut := true; + break + end + end; + if LogFileName <> '' then e_InitLog(LogFileName, TWriteMode.WM_NEWFILE); e_InitWritelnDriver(); + e_WriteLog('Doom 2D: Forever version ' + GAME_VERSION + ' proto ' + IntToStr(NET_PROTOCOL_VER), TMsgType.Notify); + e_WriteLog('Build date: ' + GAME_BUILDDATE + ' ' + GAME_BUILDTIME, TMsgType.Notify); + e_WriteLog('Build hash: ' + g_GetBuildHash(), TMsgType.Notify); + e_WriteLog('Build by: ' + g_GetBuilderName(), TMsgType.Notify); -// e_InitLog(GameDir + '/' + LogFileName, TWriteMode.WM_NEWFILE); + e_LogWritefln('Force bin dir: %s', [forceBinDir], TMsgType.Notify); + e_LogWritefln('BINARY PATH: [%s]', [binPath], TMsgType.Notify); - e_WriteLog( - 'Doom 2D: Forever version ' + GAME_VERSION + - ' proto ' + IntToStr(NET_PROTOCOL_VER), - TMsgType.Notify - ); - e_WriteLog( - 'Build date: ' + GAME_BUILDDATE + ' ' + GAME_BUILDTIME, - TMsgType.Notify - ); + PrintDirs('DataDirs', DataDirs); + PrintDirs('ModelDirs', ModelDirs); + PrintDirs('MegawadDirs', MegawadDirs); + PrintDirs('MapDirs', MapDirs); + PrintDirs('WadDirs', WadDirs); - e_LogWritefln('BINARY PATH: [%s]', [binPath], TMsgType.Notify); + PrintDirs('LogDirs', LogDirs); + PrintDirs('SaveDirs', SaveDirs); + PrintDirs('CacheDirs', CacheDirs); + PrintDirs('ConfigDirs', ConfigDirs); + PrintDirs('ScreenshotDirs', ScreenshotDirs); + PrintDirs('MapDownloadDirs', MapDownloadDirs); + PrintDirs('WadDownloadDirs', WadDownloadDirs); GameWAD := e_FindWad(DataDirs, 'GAME'); - assert(GameWad <> '', 'GAME.WAD not installed?'); + if GameWad = '' then + begin + e_WriteLog('GAME.WAD not installed?', TMsgType.Fatal); + {$IF DEFINED(USE_SDL2) AND NOT DEFINED(HEADLESS)} + if forceBinDir = false then + SDL_ShowSimpleMessageBox(SDL_MESSAGEBOX_ERROR, 'Doom 2D Forever', 'GAME.WAD not installed?', nil); + {$ENDIF} + e_DeinitLog; + Halt(1); + end; -{$IFDEF HEADLESS} - conbufDumpToStdOut := true; -{$ENDIF} - e_WriteToStdOut := False; //{$IFDEF HEADLESS}True;{$ELSE}False;{$ENDIF} + {$IF DEFINED(ANDROID) AND DEFINED(USE_SDLMIXER)} + timiditycfg := 'timidity.cfg'; + if e_FindResource(ConfigDirs, timiditycfg) = true then + begin + timiditycfg := ExpandFileName(timiditycfg); + SetEnvVar('TIMIDITY_CFG', timiditycfg); + e_LogWritefln('Set TIMIDITY_CFG = "%s"', [timiditycfg]); + end; + {$ENDIF} +end; +procedure Main(); +{$IFDEF ENABLE_HOLMES} + var flexloaded: Boolean; +{$ENDIF} + var s: AnsiString; +begin + InitPath; + InitPrep; e_InitInput; - sys_Init; + g_Options_SetDefault; + g_Options_SetDefaultVideo; s := CONFIG_FILENAME; if e_FindResource(ConfigDirs, s) = true then - begin - g_Options_Read(s) - end - else - begin - g_Options_SetDefault; - g_Options_SetDefaultVideo - end; + g_Options_Read(s); + g_Console_SysInit; if sys_SetDisplayMode(gScreenWidth, gScreenHeight, gBPP, gFullScreen) = False then raise Exception.Create('Failed to set videomode on startup.'); - g_Console_SysInit; e_WriteLog(gLanguage, TMsgType.Notify); g_Language_Set(gLanguage); @@ -368,6 +567,7 @@ begin if assigned(oglDeinitCB) then oglDeinitCB; {$ENDIF} + g_Console_WriteGameConfig; sys_Final; end;