2016-03-01 15:47:10 +00:00
|
|
|
/*
|
|
|
|
** d_iwad.cpp
|
|
|
|
** IWAD detection code
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright 1998-2009 Randy Heit
|
|
|
|
** Copyright 2009 CHristoph Oelckers
|
|
|
|
** All rights reserved.
|
|
|
|
**
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
** are met:
|
|
|
|
**
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
#include "d_main.h"
|
|
|
|
#include "gi.h"
|
|
|
|
#include "cmdlib.h"
|
|
|
|
#include "doomstat.h"
|
|
|
|
#include "i_system.h"
|
|
|
|
#include "w_wad.h"
|
|
|
|
#include "w_zip.h"
|
|
|
|
#include "v_palette.h"
|
|
|
|
#include "m_argv.h"
|
|
|
|
#include "m_misc.h"
|
|
|
|
#include "c_cvars.h"
|
|
|
|
#include "sc_man.h"
|
|
|
|
#include "v_video.h"
|
|
|
|
#include "gameconfigfile.h"
|
|
|
|
#include "resourcefiles/resourcefile.h"
|
|
|
|
#include "version.h"
|
2017-08-19 17:30:48 +00:00
|
|
|
#include "doomerrors.h"
|
|
|
|
#include "v_text.h"
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
CVAR (Bool, queryiwad, true, CVAR_ARCHIVE|CVAR_GLOBALCONFIG);
|
|
|
|
CVAR (String, defaultiwad, "", CVAR_ARCHIVE|CVAR_GLOBALCONFIG);
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Parses IWAD definitions
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
void FIWadManager::ParseIWadInfo(const char *fn, const char *data, int datasize, FIWADInfo *result)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
FScanner sc;
|
2017-08-19 17:30:48 +00:00
|
|
|
int numblocks = 0;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
sc.OpenMem("IWADINFO", data, datasize);
|
|
|
|
while (sc.GetString())
|
|
|
|
{
|
|
|
|
if (sc.Compare("IWAD"))
|
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
numblocks++;
|
|
|
|
if (result && numblocks > 1)
|
|
|
|
{
|
|
|
|
sc.ScriptMessage("Multiple IWAD records ignored");
|
|
|
|
// Skip the rest.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
FIWADInfo *iwad = result ? result : &mIWadInfos[mIWadInfos.Reserve(1)];
|
2016-03-01 15:47:10 +00:00
|
|
|
sc.MustGetStringName("{");
|
|
|
|
while (!sc.CheckString("}"))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("Name"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetString();
|
|
|
|
iwad->Name = sc.String;
|
|
|
|
}
|
|
|
|
else if (sc.Compare("Autoname"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetString();
|
|
|
|
iwad->Autoname = sc.String;
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
else if (sc.Compare("IWadname"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetString();
|
|
|
|
iwad->IWadname = sc.String;
|
|
|
|
if (sc.CheckString(","))
|
|
|
|
{
|
|
|
|
sc.MustGetNumber();
|
|
|
|
iwad->prio = sc.Number;
|
|
|
|
}
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
else if (sc.Compare("Config"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetString();
|
|
|
|
iwad->Configname = sc.String;
|
|
|
|
}
|
|
|
|
else if (sc.Compare("Game"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("Doom")) iwad->gametype = GAME_Doom;
|
|
|
|
else if (sc.Compare("Heretic")) iwad->gametype = GAME_Heretic;
|
|
|
|
else if (sc.Compare("Hexen")) iwad->gametype = GAME_Hexen;
|
|
|
|
else if (sc.Compare("Strife")) iwad->gametype = GAME_Strife;
|
|
|
|
else if (sc.Compare("Chex")) iwad->gametype = GAME_Chex;
|
|
|
|
else sc.ScriptError(NULL);
|
|
|
|
}
|
|
|
|
else if (sc.Compare("Mapinfo"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetString();
|
|
|
|
iwad->MapInfo = sc.String;
|
|
|
|
}
|
|
|
|
else if (sc.Compare("Compatibility"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
do
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if(sc.Compare("NoTextcolor")) iwad->flags |= GI_NOTEXTCOLOR;
|
|
|
|
else if(sc.Compare("Poly1")) iwad->flags |= GI_COMPATPOLY1;
|
|
|
|
else if(sc.Compare("Poly2")) iwad->flags |= GI_COMPATPOLY2;
|
|
|
|
else if(sc.Compare("Shareware")) iwad->flags |= GI_SHAREWARE;
|
|
|
|
else if(sc.Compare("Teaser2")) iwad->flags |= GI_TEASER2;
|
|
|
|
else if(sc.Compare("Extended")) iwad->flags |= GI_MENUHACK_EXTENDED;
|
|
|
|
else if(sc.Compare("Shorttex")) iwad->flags |= GI_COMPATSHORTTEX;
|
|
|
|
else if(sc.Compare("Stairs")) iwad->flags |= GI_COMPATSTAIRS;
|
|
|
|
else sc.ScriptError(NULL);
|
|
|
|
}
|
|
|
|
while (sc.CheckString(","));
|
|
|
|
}
|
|
|
|
else if (sc.Compare("MustContain"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
do
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
iwad->Lumps.Push(FString(sc.String));
|
|
|
|
}
|
|
|
|
while (sc.CheckString(","));
|
|
|
|
}
|
|
|
|
else if (sc.Compare("BannerColors"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetString();
|
2016-12-03 15:27:53 +00:00
|
|
|
iwad->FgColor = V_GetColor(NULL, sc);
|
2016-03-01 15:47:10 +00:00
|
|
|
sc.MustGetStringName(",");
|
|
|
|
sc.MustGetString();
|
2016-12-03 15:27:53 +00:00
|
|
|
iwad->BkColor = V_GetColor(NULL, sc);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
- allow the language table to supersede the title patches, if appropriate
For the Doom IWADs the provided font looks almost identical to the characters used on the title patches. So, for any level name that got replaced in some language, it will now check if the retrieved name comes from the default table, and if not, ignore the title patch and print the name with the specified font.
This also required removing the 'en' label from the default table, because with this present, the text would always be picked from 'en' instead of 'default'. Since 'en' and 'default' had the same contents, in any English locale the 'default' table was never hit, so this won't make any difference for the texts being chosen.
Last but not least, wminfo has been made a local variable in G_DoCompleted. There were two places where this was accessed from outside the summary screen or its setup code, and both were incorrect.
# Conflicts:
# src/g_hub.cpp
# src/g_level.cpp
# src/gamedata/g_mapinfo.h
# src/gi.h
# src/p_setup.cpp
# src/stringtable.cpp
# src/stringtable.h
# wadsrc/static/zscript/ui/statscreen/statscreen.zs
# wadsrc_extra/static/iwadinfo.txt
# Conflicts:
# src/gi.h
# wadsrc_extra/static/iwadinfo.txt
2019-02-14 23:29:24 +00:00
|
|
|
else if (sc.Compare("IgnoreTitlePatches"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetNumber();
|
|
|
|
if (sc.Number) iwad->flags |= GI_IGNORETITLEPATCHES;
|
|
|
|
else iwad->flags &= ~GI_IGNORETITLEPATCHES;
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
else if (sc.Compare("Load"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
do
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
iwad->Load.Push(FString(sc.String));
|
|
|
|
}
|
|
|
|
while (sc.CheckString(","));
|
|
|
|
}
|
|
|
|
else if (sc.Compare("Required"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetString();
|
|
|
|
iwad->Required = sc.String;
|
|
|
|
}
|
2017-10-14 21:41:43 +00:00
|
|
|
else if (sc.Compare("StartupType"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetString();
|
|
|
|
FString sttype = sc.String;
|
|
|
|
if (!sttype.CompareNoCase("DOOM"))
|
|
|
|
iwad->StartupType = FStartupInfo::DoomStartup;
|
|
|
|
else if (!sttype.CompareNoCase("HERETIC"))
|
|
|
|
iwad->StartupType = FStartupInfo::HereticStartup;
|
|
|
|
else if (!sttype.CompareNoCase("HEXEN"))
|
|
|
|
iwad->StartupType = FStartupInfo::HexenStartup;
|
|
|
|
else if (!sttype.CompareNoCase("STRIFE"))
|
|
|
|
iwad->StartupType = FStartupInfo::StrifeStartup;
|
|
|
|
else iwad->StartupType = FStartupInfo::DefaultStartup;
|
|
|
|
}
|
2018-04-27 10:32:38 +00:00
|
|
|
else if (sc.Compare("StartupSong"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
sc.MustGetString();
|
|
|
|
iwad->Song = sc.String;
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
sc.ScriptError("Unknown keyword '%s'", sc.String);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (iwad->MapInfo.IsEmpty())
|
|
|
|
{
|
|
|
|
// We must at least load the minimum defaults to allow the engine to run.
|
|
|
|
iwad->MapInfo = "mapinfo/mindefaults.txt";
|
|
|
|
}
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
else if (result == nullptr && sc.Compare("NAMES"))
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
sc.MustGetStringName("{");
|
|
|
|
mIWadNames.Push(FString());
|
|
|
|
while (!sc.CheckString("}"))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
FString wadname = sc.String;
|
|
|
|
mIWadNames.Push(wadname);
|
|
|
|
}
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
else if (result == nullptr && sc.Compare("ORDER"))
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("{");
|
|
|
|
while (!sc.CheckString("}"))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
mOrderNames.Push(sc.String);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sc.ScriptError("Unknown keyword '%s'", sc.String);
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Look for IWAD definition lump
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2019-07-16 18:22:15 +00:00
|
|
|
FIWadManager::FIWadManager(const char *fn, const char *optfn)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2019-07-16 18:22:15 +00:00
|
|
|
FResourceFile *resfile = FResourceFile::OpenResourceFile(optfn, true);
|
2019-07-16 19:53:53 +00:00
|
|
|
if (resfile != NULL)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-03-08 17:50:37 +00:00
|
|
|
uint32_t cnt = resfile->LumpCount();
|
2016-03-01 15:47:10 +00:00
|
|
|
for(int i=cnt-1; i>=0; i--)
|
|
|
|
{
|
|
|
|
FResourceLump *lmp = resfile->GetLump(i);
|
|
|
|
|
|
|
|
if (lmp->Namespace == ns_global && !stricmp(lmp->Name, "IWADINFO"))
|
|
|
|
{
|
|
|
|
// Found one!
|
2018-11-10 13:19:55 +00:00
|
|
|
ParseIWadInfo(resfile->FileName, (const char*)lmp->CacheLump(), lmp->LumpSize);
|
2016-03-01 15:47:10 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete resfile;
|
2019-07-16 18:22:15 +00:00
|
|
|
if (mIWadNames.Size() == 0 || mIWadInfos.Size() == 0)
|
|
|
|
{
|
|
|
|
I_FatalError("No IWAD definitions found");
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// ScanIWAD
|
|
|
|
//
|
|
|
|
// Scan the contents of an IWAD to determine which one it is
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
int FIWadManager::ScanIWAD (const char *iwad)
|
|
|
|
{
|
2018-03-11 11:33:46 +00:00
|
|
|
FResourceFile *iwadfile = FResourceFile::OpenResourceFile(iwad, true);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
mLumpsFound.Resize(mIWadInfos.Size());
|
|
|
|
|
|
|
|
auto CheckLumpName = [=](const char *name)
|
|
|
|
{
|
|
|
|
for (unsigned i = 0; i< mIWadInfos.Size(); i++)
|
|
|
|
{
|
|
|
|
for (unsigned j = 0; j < mIWadInfos[i].Lumps.Size(); j++)
|
|
|
|
{
|
|
|
|
if (!mIWadInfos[i].Lumps[j].CompareNoCase(name))
|
|
|
|
{
|
|
|
|
mLumpsFound[i] |= (1 << j);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
if (iwadfile != NULL)
|
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
memset(&mLumpsFound[0], 0, mLumpsFound.Size() * sizeof(mLumpsFound[0]));
|
2017-03-08 17:50:37 +00:00
|
|
|
for(uint32_t ii = 0; ii < iwadfile->LumpCount(); ii++)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
FResourceLump *lump = iwadfile->GetLump(ii);
|
|
|
|
|
|
|
|
CheckLumpName(lump->Name);
|
|
|
|
if (lump->FullName.IsNotEmpty())
|
|
|
|
{
|
|
|
|
if (strnicmp(lump->FullName, "maps/", 5) == 0)
|
|
|
|
{
|
|
|
|
FString mapname(&lump->FullName[5], strcspn(&lump->FullName[5], "."));
|
|
|
|
CheckLumpName(mapname);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete iwadfile;
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
for (unsigned i = 0; i< mIWadInfos.Size(); i++)
|
|
|
|
{
|
|
|
|
if (mLumpsFound[i] == (1 << mIWadInfos[i].Lumps.Size()) - 1)
|
|
|
|
{
|
|
|
|
DPrintf(DMSG_NOTIFY, "Identified %s as %s\n", iwad, mIWadInfos[i].Name.GetChars());
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
2017-08-19 17:30:48 +00:00
|
|
|
// Look for IWAD definition lump
|
2016-03-01 15:47:10 +00:00
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
int FIWadManager::CheckIWADInfo(const char *fn)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2018-03-11 11:33:46 +00:00
|
|
|
FResourceFile *resfile = FResourceFile::OpenResourceFile(fn, true);
|
2017-08-19 17:30:48 +00:00
|
|
|
if (resfile != NULL)
|
|
|
|
{
|
|
|
|
uint32_t cnt = resfile->LumpCount();
|
|
|
|
for (int i = cnt - 1; i >= 0; i--)
|
|
|
|
{
|
|
|
|
FResourceLump *lmp = resfile->GetLump(i);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
if (lmp->Namespace == ns_global && !stricmp(lmp->Name, "IWADINFO"))
|
|
|
|
{
|
|
|
|
// Found one!
|
|
|
|
try
|
|
|
|
{
|
|
|
|
FIWADInfo result;
|
2018-11-10 13:19:55 +00:00
|
|
|
ParseIWadInfo(resfile->FileName, (const char*)lmp->CacheLump(), lmp->LumpSize, &result);
|
2017-08-19 17:30:48 +00:00
|
|
|
delete resfile;
|
2018-04-02 13:16:47 +00:00
|
|
|
|
|
|
|
for (unsigned i = 0, count = mIWadInfos.Size(); i < count; ++i)
|
|
|
|
{
|
|
|
|
if (mIWadInfos[i].Name == result.Name)
|
|
|
|
{
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mOrderNames.Push(result.Name);
|
2017-08-19 17:30:48 +00:00
|
|
|
return mIWadInfos.Push(result);
|
|
|
|
}
|
|
|
|
catch (CRecoverableError &err)
|
|
|
|
{
|
|
|
|
delete resfile;
|
|
|
|
Printf(TEXTCOLOR_RED "%s: %s\nFile has been removed from the list of IWADs\n", fn, err.GetMessage());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete resfile;
|
|
|
|
Printf(TEXTCOLOR_RED "%s: Unable to find IWADINFO\nFile has been removed from the list of IWADs\n", fn);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
Printf(TEXTCOLOR_RED "%s: Unable to open as resource file.\nFile has been removed from the list of IWADs\n", fn);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// CollectSearchPaths
|
|
|
|
//
|
|
|
|
// collect all paths in a local array for easier management
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void FIWadManager::CollectSearchPaths()
|
|
|
|
{
|
|
|
|
if (GameConfig->SetSection("IWADSearch.Directories"))
|
|
|
|
{
|
|
|
|
const char *key;
|
|
|
|
const char *value;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
while (GameConfig->NextInSection(key, value))
|
|
|
|
{
|
|
|
|
if (stricmp(key, "Path") == 0)
|
|
|
|
{
|
|
|
|
FString nice = NicePath(value);
|
|
|
|
if (nice.Len() > 0) mSearchPaths.Push(nice);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mSearchPaths.Append(I_GetGogPaths());
|
|
|
|
mSearchPaths.Append(I_GetSteamPath());
|
|
|
|
|
|
|
|
// Unify and remove trailing slashes
|
|
|
|
for (auto &str : mSearchPaths)
|
|
|
|
{
|
|
|
|
FixPathSeperator(str);
|
2017-10-23 08:55:20 +00:00
|
|
|
if (str.Back() == '/') str.Truncate(str.Len() - 1);
|
2017-08-19 17:30:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// AddIWADCandidates
|
|
|
|
//
|
|
|
|
// scans the given directory and adds all potential IWAD candidates
|
|
|
|
//
|
|
|
|
//==========================================================================
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
void FIWadManager::AddIWADCandidates(const char *dir)
|
|
|
|
{
|
|
|
|
void *handle;
|
|
|
|
findstate_t findstate;
|
|
|
|
FStringf slasheddir("%s/", dir);
|
|
|
|
FString findmask = slasheddir + "*.*";
|
|
|
|
if ((handle = I_FindFirst(findmask, &findstate)) != (void *)-1)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
do
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
if (!(I_FindAttr(&findstate) & FA_DIREC))
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-20 06:01:41 +00:00
|
|
|
auto FindName = I_FindName(&findstate);
|
|
|
|
auto p = strrchr(FindName, '.');
|
2017-08-19 17:30:48 +00:00
|
|
|
if (p != nullptr)
|
|
|
|
{
|
|
|
|
// special IWAD extension.
|
2018-04-02 12:32:00 +00:00
|
|
|
if (!stricmp(p, ".iwad") || !stricmp(p, ".ipk3") || !stricmp(p, ".ipk7"))
|
2017-08-19 17:30:48 +00:00
|
|
|
{
|
2017-08-20 06:01:41 +00:00
|
|
|
mFoundWads.Push(FFoundWadInfo{ slasheddir + FindName, "", -1 });
|
2017-08-19 17:30:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
for (auto &name : mIWadNames)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-20 06:01:41 +00:00
|
|
|
if (!stricmp(name, FindName))
|
2017-08-19 17:30:48 +00:00
|
|
|
{
|
2017-09-05 07:33:07 +00:00
|
|
|
mFoundWads.Push(FFoundWadInfo{ slasheddir + FindName, "", -1 });
|
2017-08-19 17:30:48 +00:00
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
} while (I_FindNext(handle, &findstate) == 0);
|
|
|
|
I_FindClose(handle);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// ValidateIWADs
|
|
|
|
//
|
|
|
|
// validate all found candidates and eliminate the bogus ones.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void FIWadManager::ValidateIWADs()
|
|
|
|
{
|
|
|
|
TArray<int> returns;
|
|
|
|
unsigned originalsize = mIWadInfos.Size();
|
|
|
|
for (auto &p : mFoundWads)
|
|
|
|
{
|
|
|
|
int index;
|
|
|
|
auto x = strrchr(p.mFullPath, '.');
|
2018-04-02 12:32:00 +00:00
|
|
|
if (x != nullptr && (!stricmp(x, ".iwad") || !stricmp(x, ".ipk3") || !stricmp(x, ".ipk7")))
|
2017-08-19 17:30:48 +00:00
|
|
|
{
|
|
|
|
index = CheckIWADInfo(p.mFullPath);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
index = ScanIWAD(p.mFullPath);
|
|
|
|
}
|
|
|
|
p.mInfoIndex = index;
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// IdentifyVersion
|
|
|
|
//
|
|
|
|
// Tries to find an IWAD in one of four directories under DOS or Win32:
|
|
|
|
// 1. Current directory
|
|
|
|
// 2. Executable directory
|
|
|
|
// 3. $DOOMWADDIR
|
|
|
|
// 4. $HOME
|
|
|
|
//
|
|
|
|
// Under UNIX OSes, the search path is:
|
|
|
|
// 1. Current directory
|
|
|
|
// 2. $DOOMWADDIR
|
|
|
|
// 3. $HOME/.config/zdoom
|
|
|
|
// 4. The share directory defined at compile time (/usr/local/share/zdoom)
|
|
|
|
//
|
|
|
|
// The search path can be altered by editing the IWADSearch.Directories
|
|
|
|
// section of the config file.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-03-08 13:35:36 +00:00
|
|
|
static bool havepicked = false;
|
|
|
|
|
2017-09-09 10:36:20 +00:00
|
|
|
int FIWadManager::IdentifyVersion (TArray<FString> &wadfiles, const char *iwad, const char *zdoom_wad, const char *optional_wad)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
const char *iwadparm = Args->CheckValue ("-iwad");
|
|
|
|
FString custwad;
|
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
CollectSearchPaths();
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
// Collect all IWADs in the search path
|
|
|
|
for (auto &dir : mSearchPaths)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
AddIWADCandidates(dir);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
unsigned numFoundWads = mFoundWads.Size();
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
if (iwadparm)
|
|
|
|
{
|
2018-03-31 12:20:00 +00:00
|
|
|
const char* const extensions[] = { ".wad", ".pk3", ".iwad", ".ipk3", ".ipk7" };
|
|
|
|
|
|
|
|
for (auto ext : extensions)
|
|
|
|
{
|
|
|
|
// Check if the given IWAD has an absolute path, in which case the search path will be ignored.
|
|
|
|
custwad = iwadparm;
|
|
|
|
FixPathSeperator(custwad);
|
|
|
|
DefaultExtension(custwad, ext);
|
|
|
|
bool isAbsolute = (custwad[0] == '/');
|
2018-05-12 09:32:53 +00:00
|
|
|
#ifdef _WIN32
|
2018-03-31 12:20:00 +00:00
|
|
|
isAbsolute |= (custwad.Len() >= 2 && custwad[1] == ':');
|
2017-08-19 17:30:48 +00:00
|
|
|
#endif
|
2018-03-31 12:20:00 +00:00
|
|
|
if (isAbsolute)
|
|
|
|
{
|
|
|
|
if (FileExists(custwad)) mFoundWads.Push({ custwad, "", -1 });
|
|
|
|
}
|
|
|
|
else
|
2017-08-19 17:30:48 +00:00
|
|
|
{
|
2018-03-31 12:20:00 +00:00
|
|
|
for (auto &dir : mSearchPaths)
|
2017-08-19 17:30:48 +00:00
|
|
|
{
|
2018-03-31 12:20:00 +00:00
|
|
|
FStringf fullpath("%s/%s", dir.GetChars(), custwad.GetChars());
|
|
|
|
if (FileExists(fullpath))
|
|
|
|
{
|
|
|
|
mFoundWads.Push({ fullpath, "", -1 });
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
}
|
|
|
|
}
|
2018-03-31 12:20:00 +00:00
|
|
|
|
|
|
|
if (mFoundWads.Size() != numFoundWads)
|
|
|
|
{
|
|
|
|
// Found IWAD with guessed extension
|
|
|
|
break;
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
// -iwad not found or not specified. Revert back to standard behavior.
|
|
|
|
if (mFoundWads.Size() == numFoundWads) iwadparm = nullptr;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
// Now check if what got collected actually is an IWAD.
|
|
|
|
ValidateIWADs();
|
|
|
|
|
|
|
|
// Check for required dependencies.
|
|
|
|
for (unsigned i = 0; i < mFoundWads.Size(); i++)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
auto infndx = mFoundWads[i].mInfoIndex;
|
|
|
|
if (infndx >= 0)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
auto &wadinfo = mIWadInfos[infndx];
|
|
|
|
if (wadinfo.Required.IsNotEmpty())
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
bool found = false;
|
|
|
|
// needs to be loaded with another IWAD (HexenDK)
|
|
|
|
for (unsigned j = 0; j < mFoundWads.Size(); j++)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
auto inf2ndx = mFoundWads[j].mInfoIndex;
|
|
|
|
if (inf2ndx >= 0)
|
|
|
|
{
|
|
|
|
if (!mIWadInfos[infndx].Required.Compare(mIWadInfos[inf2ndx].Name))
|
|
|
|
{
|
|
|
|
mFoundWads[i].mRequiredPath = mFoundWads[j].mFullPath;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
// The required dependency was not found. Skip this IWAD.
|
|
|
|
if (mFoundWads[i].mRequiredPath.IsEmpty()) mFoundWads[i].mInfoIndex = -1;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
TArray<FFoundWadInfo> picks;
|
|
|
|
if (numFoundWads < mFoundWads.Size())
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
// We have a -iwad parameter. Pick the first usable IWAD we found through that.
|
|
|
|
for (unsigned i = numFoundWads; i < mFoundWads.Size(); i++)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-11-18 12:35:45 +00:00
|
|
|
if (mFoundWads[i].mInfoIndex >= 0)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
picks.Push(mFoundWads[i]);
|
|
|
|
break;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
else if (iwad != nullptr && *iwad != 0)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
int pickedprio = -1;
|
|
|
|
// scan the list of found IWADs for a matching one for the current PWAD.
|
|
|
|
for (auto &found : mFoundWads)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-09-13 22:23:53 +00:00
|
|
|
if (found.mInfoIndex >= 0 && mIWadInfos[found.mInfoIndex].IWadname.CompareNoCase(iwad) == 0 && mIWadInfos[found.mInfoIndex].prio > pickedprio)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
picks.Clear();
|
|
|
|
picks.Push(found);
|
|
|
|
pickedprio = mIWadInfos[found.mInfoIndex].prio;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (picks.Size() == 0)
|
|
|
|
{
|
|
|
|
// Now sort what we found and discard all duplicates.
|
|
|
|
for (unsigned i = 0; i < mOrderNames.Size(); i++)
|
|
|
|
{
|
|
|
|
bool picked = false;
|
|
|
|
for (int j = 0; j < (int)mFoundWads.Size(); j++)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-27 07:57:37 +00:00
|
|
|
if (mFoundWads[j].mInfoIndex >= 0)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-27 07:57:37 +00:00
|
|
|
if (mIWadInfos[mFoundWads[j].mInfoIndex].Name.Compare(mOrderNames[i]) == 0)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-27 07:57:37 +00:00
|
|
|
if (!picked)
|
|
|
|
{
|
|
|
|
picked = true;
|
|
|
|
picks.Push(mFoundWads[j]);
|
|
|
|
}
|
|
|
|
mFoundWads.Delete(j--);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
// What's left is IWADs with their own IWADINFO. Copy those in order of discovery.
|
|
|
|
for (auto &entry : mFoundWads)
|
|
|
|
{
|
|
|
|
if (entry.mInfoIndex >= 0) picks.Push(entry);
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
// If we still haven't found a suitable IWAD let's error out.
|
|
|
|
if (picks.Size() == 0)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
I_FatalError ("Cannot find a game IWAD (doom.wad, doom2.wad, heretic.wad, etc.).\n"
|
|
|
|
"Did you install " GAMENAME " properly? You can do either of the following:\n"
|
|
|
|
"\n"
|
|
|
|
#if defined(_WIN32)
|
|
|
|
"1. Place one or more of these wads in the same directory as " GAMENAME ".\n"
|
|
|
|
"2. Edit your " GAMENAMELOWERCASE "-username.ini and add the directories of your iwads\n"
|
|
|
|
"to the list beneath [IWADSearch.Directories]");
|
|
|
|
#elif defined(__APPLE__)
|
|
|
|
"1. Place one or more of these wads in ~/Library/Application Support/" GAMENAMELOWERCASE "/\n"
|
|
|
|
"2. Edit your ~/Library/Preferences/" GAMENAMELOWERCASE ".ini and add the directories\n"
|
|
|
|
"of your iwads to the list beneath [IWADSearch.Directories]");
|
|
|
|
#else
|
|
|
|
"1. Place one or more of these wads in ~/.config/" GAMENAMELOWERCASE "/.\n"
|
|
|
|
"2. Edit your ~/.config/" GAMENAMELOWERCASE "/" GAMENAMELOWERCASE ".ini and add the directories of your\n"
|
|
|
|
"iwads to the list beneath [IWADSearch.Directories]");
|
|
|
|
#endif
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
int pick = 0;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
// We got more than one so present the IWAD selection box.
|
|
|
|
if (picks.Size() > 1)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
// Locate the user's prefered IWAD, if it was found.
|
|
|
|
if (defaultiwad[0] != '\0')
|
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
for (unsigned i = 0; i < picks.Size(); ++i)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-20 07:27:32 +00:00
|
|
|
FString &basename = mIWadInfos[picks[i].mInfoIndex].Name;
|
|
|
|
if (stricmp(basename, defaultiwad) == 0)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
pick = i;
|
2016-03-01 15:47:10 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
if (picks.Size() > 1)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
if (!havepicked)
|
2016-03-08 13:35:36 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
TArray<WadStuff> wads;
|
|
|
|
for (auto & found : picks)
|
|
|
|
{
|
|
|
|
WadStuff stuff;
|
|
|
|
stuff.Name = mIWadInfos[found.mInfoIndex].Name;
|
|
|
|
stuff.Path = ExtractFileBase(found.mFullPath);
|
|
|
|
wads.Push(stuff);
|
|
|
|
}
|
|
|
|
pick = I_PickIWad(&wads[0], (int)wads.Size(), queryiwad, pick);
|
|
|
|
if (pick >= 0)
|
|
|
|
{
|
|
|
|
// The newly selected IWAD becomes the new default
|
|
|
|
defaultiwad = mIWadInfos[picks[pick].mInfoIndex].Name;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
havepicked = true;
|
2016-03-08 13:35:36 +00:00
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// zdoom.pk3 must always be the first file loaded and the IWAD second.
|
|
|
|
wadfiles.Clear();
|
|
|
|
D_AddFile (wadfiles, zdoom_wad);
|
|
|
|
|
2017-09-09 10:21:40 +00:00
|
|
|
// [SP] Load non-free assets if available. This must be done before the IWAD.
|
2017-09-13 08:19:03 +00:00
|
|
|
if (D_AddFile(wadfiles, optional_wad))
|
|
|
|
Wads.SetIwadNum(2);
|
|
|
|
else
|
|
|
|
Wads.SetIwadNum(1);
|
2017-09-09 10:21:40 +00:00
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
if (picks[pick].mRequiredPath.IsNotEmpty())
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
D_AddFile (wadfiles, picks[pick].mRequiredPath);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
D_AddFile (wadfiles, picks[pick].mFullPath);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-08-19 17:30:48 +00:00
|
|
|
auto info = mIWadInfos[picks[pick].mInfoIndex];
|
|
|
|
// Load additional resources from the same directory as the IWAD itself.
|
|
|
|
for (unsigned i=0; i < info.Load.Size(); i++)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
long lastslash = picks[pick].mFullPath.LastIndexOf ('/');
|
2016-03-01 15:47:10 +00:00
|
|
|
FString path;
|
|
|
|
|
|
|
|
if (lastslash == -1)
|
|
|
|
{
|
|
|
|
path = "";// wads[pickwad].Path;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-08-19 17:30:48 +00:00
|
|
|
path = FString (picks[pick].mFullPath.GetChars(), lastslash + 1);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
path += info.Load[i];
|
2016-03-01 15:47:10 +00:00
|
|
|
D_AddFile (wadfiles, path);
|
|
|
|
|
|
|
|
}
|
2017-08-19 17:30:48 +00:00
|
|
|
return picks[pick].mInfoIndex;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Find an IWAD to use for this game
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2017-09-09 10:36:20 +00:00
|
|
|
const FIWADInfo *FIWadManager::FindIWAD(TArray<FString> &wadfiles, const char *iwad, const char *basewad, const char *optionalwad)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-09-09 10:36:20 +00:00
|
|
|
int iwadType = IdentifyVersion(wadfiles, iwad, basewad, optionalwad);
|
2016-03-01 15:47:10 +00:00
|
|
|
//gameiwad = iwadType;
|
2017-08-19 17:30:48 +00:00
|
|
|
const FIWADInfo *iwad_info = &mIWadInfos[iwadType];
|
2018-07-20 10:33:49 +00:00
|
|
|
if (DoomStartupInfo.Name.IsEmpty())
|
|
|
|
{
|
|
|
|
DoomStartupInfo.Name = iwad_info->Name;
|
|
|
|
if (iwad_info->gametype == GAME_Doom)
|
|
|
|
DoomStartupInfo.Name << " v" << VERSIONSTR;
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
if (DoomStartupInfo.BkColor == 0 && DoomStartupInfo.FgColor == 0)
|
|
|
|
{
|
|
|
|
DoomStartupInfo.BkColor = iwad_info->BkColor;
|
|
|
|
DoomStartupInfo.FgColor = iwad_info->FgColor;
|
|
|
|
}
|
2017-10-14 21:41:43 +00:00
|
|
|
if (DoomStartupInfo.Type == 0) DoomStartupInfo.Type = iwad_info->StartupType;
|
2018-04-27 10:32:38 +00:00
|
|
|
if (DoomStartupInfo.Song.IsEmpty()) DoomStartupInfo.Song = iwad_info->Song;
|
2016-03-01 15:47:10 +00:00
|
|
|
I_SetIWADInfo();
|
|
|
|
return iwad_info;
|
2017-09-05 07:33:07 +00:00
|
|
|
}
|