Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
hrydgard
GitHub Repository: hrydgard/ppsspp
Path: blob/master/Core/ELF/ParamSFO.cpp
3186 views
1
// Copyright (c) 2012- PPSSPP Project.
2
3
// This program is free software: you can redistribute it and/or modify
4
// it under the terms of the GNU General Public License as published by
5
// the Free Software Foundation, version 2.0 or later versions.
6
7
// This program is distributed in the hope that it will be useful,
8
// but WITHOUT ANY WARRANTY; without even the implied warranty of
9
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10
// GNU General Public License 2.0 for more details.
11
12
// A copy of the GPL 2.0 should have been included with the program.
13
// If not, see http://www.gnu.org/licenses/
14
15
// Official git repository and contact information can be found at
16
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
17
18
#include <cstdio>
19
#include <cstring>
20
21
#include "Common/CommonTypes.h"
22
#include "Common/Log.h"
23
#include "Common/StringUtils.h"
24
#include "Common/Swap.h"
25
#include "Common/File/Path.h"
26
#include "Core/ELF/ParamSFO.h"
27
#include "Core/System.h"
28
29
struct Header
30
{
31
u32_le magic; /* Always PSF */
32
u32_le version; /* Usually 1.1 */
33
u32_le key_table_start; /* Start position of key_table */
34
u32_le data_table_start; /* Start position of data_table */
35
u32_le index_table_entries; /* Number of entries in index_table*/
36
};
37
38
struct IndexTable
39
{
40
u16_le key_table_offset; /* Offset of the param_key from start of key_table */
41
u16_le param_fmt; /* Type of data of param_data in the data_table */
42
u32_le param_len; /* Used Bytes by param_data in the data_table */
43
u32_le param_max_len; /* Total bytes reserved for param_data in the data_table */
44
u32_le data_table_offset; /* Offset of the param_data from start of data_table */
45
};
46
47
void ParamSFOData::SetValue(const std::string &key, unsigned int value, int max_size) {
48
values[key].type = VT_INT;
49
values[key].i_value = value;
50
values[key].max_size = max_size;
51
}
52
void ParamSFOData::SetValue(const std::string &key, const std::string &value, int max_size) {
53
values[key].type = VT_UTF8;
54
values[key].s_value = value;
55
values[key].max_size = max_size;
56
}
57
58
void ParamSFOData::SetValue(const std::string &key, const u8 *value, unsigned int size, int max_size) {
59
values[key].type = VT_UTF8_SPE;
60
values[key].SetData(value, size);
61
values[key].max_size = max_size;
62
}
63
64
int ParamSFOData::GetValueInt(std::string_view key) const {
65
std::map<std::string,ValueData>::const_iterator it = values.find(key);
66
if (it == values.end() || it->second.type != VT_INT)
67
return 0;
68
return it->second.i_value;
69
}
70
71
std::string ParamSFOData::GetValueString(std::string_view key) const {
72
std::map<std::string,ValueData>::const_iterator it = values.find(key);
73
if (it == values.end() || (it->second.type != VT_UTF8))
74
return "";
75
return it->second.s_value;
76
}
77
78
bool ParamSFOData::HasKey(std::string_view key) const {
79
return values.find(key) != values.end();
80
}
81
82
const u8 *ParamSFOData::GetValueData(std::string_view key, unsigned int *size) const {
83
std::map<std::string,ValueData>::const_iterator it = values.find(key);
84
if (it == values.end() || (it->second.type != VT_UTF8_SPE)) {
85
return 0;
86
}
87
if (size) {
88
*size = it->second.u_size;
89
}
90
return it->second.u_value;
91
}
92
93
std::vector<std::string> ParamSFOData::GetKeys() const {
94
std::vector<std::string> result;
95
for (const auto &pair : values) {
96
result.push_back(pair.first);
97
}
98
return result;
99
}
100
101
std::string ParamSFOData::GetDiscID() {
102
const std::string discID = GetValueString("DISC_ID");
103
if (discID.empty()) {
104
std::string fakeID = GenerateFakeID(Path());
105
WARN_LOG(Log::Loader, "No DiscID found - generating a fake one: '%s' (from %s)", fakeID.c_str(), PSP_CoreParameter().fileToStart.c_str());
106
ValueData data;
107
data.type = VT_UTF8;
108
data.s_value = fakeID;
109
values["DISC_ID"] = data;
110
return fakeID;
111
}
112
return discID;
113
}
114
115
// I'm so sorry Ced but this is highly endian unsafe :(
116
bool ParamSFOData::ReadSFO(const u8 *paramsfo, size_t size) {
117
if (size < sizeof(Header))
118
return false;
119
const Header *header = (const Header *)paramsfo;
120
if (header->magic != 0x46535000)
121
return false;
122
if (header->version != 0x00000101)
123
WARN_LOG(Log::Loader, "Unexpected SFO header version: %08x", header->version);
124
125
const IndexTable *indexTables = (const IndexTable *)(paramsfo + sizeof(Header));
126
127
if (header->key_table_start > size || header->data_table_start > size) {
128
return false;
129
}
130
131
const u8 *data_start = paramsfo + header->data_table_start;
132
133
auto readStringCapped = [paramsfo, size](size_t offset, size_t maxLen) -> std::string {
134
std::string str;
135
while (offset < size) {
136
char c = (char)(paramsfo[offset]);
137
if (c) {
138
str.push_back(c);
139
} else {
140
break;
141
}
142
offset++;
143
if (maxLen != 0 && str.size() == maxLen)
144
break;
145
}
146
return str;
147
};
148
149
for (u32 i = 0; i < header->index_table_entries; i++)
150
{
151
size_t key_offset = header->key_table_start + indexTables[i].key_table_offset;
152
if (key_offset >= size) {
153
return false;
154
}
155
156
size_t data_offset = header->data_table_start + indexTables[i].data_table_offset;
157
if (data_offset >= size) {
158
return false;
159
}
160
161
std::string key = readStringCapped(key_offset, 0);
162
if (key.empty())
163
continue; // Likely ran into a truncated PARAMSFO.
164
165
switch (indexTables[i].param_fmt) {
166
case 0x0404:
167
{
168
if (data_offset + 4 > size)
169
continue;
170
// Unsigned int
171
const u32_le *data = (const u32_le *)(paramsfo + data_offset);
172
SetValue(key, *data, indexTables[i].param_max_len);
173
VERBOSE_LOG(Log::Loader, "%s %08x", key.c_str(), *data);
174
}
175
break;
176
case 0x0004:
177
// Special format UTF-8
178
{
179
if (data_offset + indexTables[i].param_len > size)
180
continue;
181
const u8 *utfdata = (const u8 *)(paramsfo + data_offset);
182
VERBOSE_LOG(Log::Loader, "%s %s", key.c_str(), utfdata);
183
SetValue(key, utfdata, indexTables[i].param_len, indexTables[i].param_max_len);
184
}
185
break;
186
case 0x0204:
187
// Regular UTF-8
188
{
189
// TODO: Likely should use param_len here, but there's gotta be a reason we avoided it before.
190
std::string str = readStringCapped(data_offset, indexTables[i].param_max_len);
191
VERBOSE_LOG(Log::Loader, "%s %s", key.c_str(), str.c_str());
192
SetValue(key, str, indexTables[i].param_max_len);
193
}
194
break;
195
default:
196
break;
197
}
198
}
199
200
return true;
201
}
202
203
int ParamSFOData::GetDataOffset(const u8 *paramsfo, const char *dataName) {
204
const Header *header = (const Header *)paramsfo;
205
if (header->magic != 0x46535000)
206
return -1;
207
if (header->version != 0x00000101)
208
WARN_LOG(Log::Loader, "Unexpected SFO header version: %08x", header->version);
209
210
const IndexTable *indexTables = (const IndexTable *)(paramsfo + sizeof(Header));
211
212
const u8 *key_start = paramsfo + header->key_table_start;
213
int data_start = header->data_table_start;
214
215
for (u32 i = 0; i < header->index_table_entries; i++)
216
{
217
const char *key = (const char *)(key_start + indexTables[i].key_table_offset);
218
if (!strcmp(key, dataName))
219
{
220
return data_start + indexTables[i].data_table_offset;
221
}
222
}
223
224
return -1;
225
}
226
227
void ParamSFOData::WriteSFO(u8 **paramsfo, size_t *size) const {
228
size_t total_size = 0;
229
size_t key_size = 0;
230
size_t data_size = 0;
231
232
Header header;
233
header.magic = 0x46535000;
234
header.version = 0x00000101;
235
header.index_table_entries = 0;
236
237
total_size += sizeof(Header);
238
239
// Get size info
240
for (const auto &[k, v] : values)
241
{
242
key_size += k.size() + 1;
243
data_size += v.max_size;
244
245
header.index_table_entries++;
246
}
247
248
// Padding
249
while ((key_size % 4) != 0) key_size++;
250
251
header.key_table_start = sizeof(Header) + header.index_table_entries * sizeof(IndexTable);
252
header.data_table_start = header.key_table_start + (u32)key_size;
253
254
total_size += sizeof(IndexTable) * header.index_table_entries;
255
total_size += key_size;
256
total_size += data_size;
257
*size = total_size;
258
259
size_t aligned_size = (total_size + 15) & ~15;
260
u8* data = new u8[aligned_size];
261
*paramsfo = data;
262
memset(data, 0, aligned_size);
263
memcpy(data, &header, sizeof(Header));
264
265
// Now fill
266
IndexTable *index_ptr = (IndexTable*)(data + sizeof(Header));
267
u8* key_ptr = data + header.key_table_start;
268
u8* data_ptr = data + header.data_table_start;
269
270
for (const auto &[k, v] : values)
271
{
272
u16 offset = (u16)(key_ptr - (data+header.key_table_start));
273
index_ptr->key_table_offset = offset;
274
offset = (u16)(data_ptr - (data+header.data_table_start));
275
index_ptr->data_table_offset = offset;
276
index_ptr->param_max_len = v.max_size;
277
if (v.type == VT_INT)
278
{
279
index_ptr->param_fmt = 0x0404;
280
index_ptr->param_len = 4;
281
282
*(s32_le *)data_ptr = v.i_value;
283
}
284
else if (v.type == VT_UTF8_SPE)
285
{
286
index_ptr->param_fmt = 0x0004;
287
index_ptr->param_len = v.u_size;
288
289
memset(data_ptr,0,index_ptr->param_max_len);
290
memcpy(data_ptr,v.u_value,index_ptr->param_len);
291
}
292
else if (v.type == VT_UTF8)
293
{
294
index_ptr->param_fmt = 0x0204;
295
index_ptr->param_len = (u32)v.s_value.size()+1;
296
297
memcpy(data_ptr,v.s_value.c_str(),index_ptr->param_len);
298
data_ptr[index_ptr->param_len] = 0;
299
}
300
301
memcpy(key_ptr,k.c_str(),k.size());
302
key_ptr[k.size()] = 0;
303
304
data_ptr += index_ptr->param_max_len;
305
key_ptr += k.size() + 1;
306
index_ptr++;
307
308
}
309
}
310
311
void ParamSFOData::Clear() {
312
values.clear();
313
}
314
315
void ParamSFOData::ValueData::SetData(const u8* data, int size) {
316
if (u_value) {
317
delete[] u_value;
318
u_value = 0;
319
}
320
if (size > 0) {
321
u_value = new u8[size];
322
memcpy(u_value, data, size);
323
}
324
u_size = size;
325
}
326
327
std::string ParamSFOData::GenerateFakeID(const Path &filename) const {
328
// Generates fake gameID for homebrew based on it's folder name.
329
// Should probably not be a part of ParamSFO, but it'll be called in same places.
330
// FileToStart here is actually a directory name, not a file, so taking GetFilename on it gets what we want.
331
Path path = PSP_CoreParameter().fileToStart;
332
if (!filename.empty())
333
path = filename;
334
335
std::string file = path.GetFilename();
336
337
int sumOfAllLetters = 0;
338
for (char &c : file) {
339
sumOfAllLetters += c;
340
// Get rid of some garbage characters than can arise when opening content URIs. Well, I've only seen '%', but...
341
if (strchr("%() []", c) != nullptr) {
342
c = 'X';
343
} else {
344
c = toupper(c);
345
}
346
}
347
348
if (file.size() < 4) {
349
file += "HOME";
350
}
351
file = file.substr(0, 4);
352
353
std::string fakeID = file + StringFromFormat("%05d", sumOfAllLetters);
354
return fakeID;
355
}
356
357
GameRegion DetectGameRegionFromID(std::string_view id_version) {
358
if (id_version.size() >= 4) {
359
std::string_view regStr = id_version.substr(0, 4);
360
361
// Guesswork
362
switch (regStr[2]) {
363
case 'E': return GameRegion::EUROPE; break;
364
case 'U': return GameRegion::USA; break;
365
case 'J': return GameRegion::JAPAN; break;
366
case 'H': return GameRegion::HONGKONG; break;
367
case 'A': return GameRegion::ASIA; break;
368
case 'K': return GameRegion::KOREA; break;
369
default: return GameRegion::OTHER;
370
}
371
/*
372
if (regStr == "NPEZ" || regStr == "NPEG" || regStr == "ULES" || regStr == "UCES" ||
373
regStr == "NPEX") {
374
region = GameRegion::EUROPE;
375
} else if (regStr == "NPUG" || regStr == "NPUZ" || regStr == "ULUS" || regStr == "UCUS") {
376
region = GameRegion::USA;
377
} else if (regStr == "NPJH" || regStr == "NPJG" || regStr == "ULJM"|| regStr == "ULJS") {
378
region = GameRegion::JAPAN;
379
} else if (regStr == "NPHG") {
380
region = GameRegion::HONGKONG;
381
} else if (regStr == "UCAS") {
382
region = GameRegion::CHINA;
383
}*/
384
} else {
385
return GameRegion::OTHER;
386
}
387
}
388
389
std::string_view GameRegionToString(GameRegion region) {
390
switch (region) {
391
case GameRegion::JAPAN: return "Japan";
392
case GameRegion::USA: return "USA";
393
case GameRegion::EUROPE: return "Europe";
394
case GameRegion::HONGKONG: return "Hong Kong";
395
case GameRegion::ASIA: return "Asia";
396
case GameRegion::KOREA: return "Korea";
397
case GameRegion::HOMEBREW: return "Homebrew";
398
default: return "Other";
399
}
400
}
401
402