| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package PlayStation::MemoryCard; |
|
2
|
|
|
|
|
|
|
# Copyright (c) 2021 Gavin Hayes and others, see LICENSE in the root of the project |
|
3
|
1
|
|
|
1
|
|
935
|
use version; our $VERSION = version->declare("v0.2.1"); |
|
|
1
|
|
|
|
|
2005
|
|
|
|
1
|
|
|
|
|
5
|
|
|
4
|
1
|
|
|
1
|
|
102
|
use strict; |
|
|
1
|
|
|
|
|
3
|
|
|
|
1
|
|
|
|
|
21
|
|
|
5
|
1
|
|
|
1
|
|
5
|
use warnings; |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
28
|
|
|
6
|
1
|
|
|
1
|
|
567
|
use Encode qw(decode encode); |
|
|
1
|
|
|
|
|
10416
|
|
|
|
1
|
|
|
|
|
93
|
|
|
7
|
1
|
|
|
1
|
|
8
|
use File::Basename; |
|
|
1
|
|
|
|
|
3
|
|
|
|
1
|
|
|
|
|
1799
|
|
|
8
|
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
sub parse_directory { |
|
10
|
0
|
|
|
0
|
0
|
|
my ($directory) = @_; |
|
11
|
0
|
|
|
|
|
|
my $inuse = unpack('C', $directory); |
|
12
|
0
|
|
|
|
|
|
my $datasize = unpack('V', substr($directory, 0x4, 0x4)); |
|
13
|
0
|
|
|
|
|
|
my $linkindex = unpack('v', substr($directory, 0x8)); |
|
14
|
0
|
|
|
|
|
|
my $codestr = unpack('Z*', substr($directory, 0xA)); |
|
15
|
0
|
|
|
|
|
|
my @toxor = unpack('C127', $directory); |
|
16
|
0
|
|
|
|
|
|
my $storedxor = unpack('C', substr($directory, 0x7F)); |
|
17
|
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
#my @blocktypes = ('INUSE', 'SRTLINK', 'MIDLINK', 'ENDLINK', 'EMPTY', 'UNUSABLE', 'UNKNOWN'); |
|
19
|
0
|
|
|
|
|
|
my $blockcount = int($datasize / 0x2000); |
|
20
|
|
|
|
|
|
|
|
|
21
|
0
|
|
|
|
|
|
my $calcxor = 0; |
|
22
|
0
|
|
|
|
|
|
foreach my $char (@toxor) { |
|
23
|
0
|
|
|
|
|
|
$calcxor ^= $char; |
|
24
|
|
|
|
|
|
|
} |
|
25
|
|
|
|
|
|
|
|
|
26
|
|
|
|
|
|
|
return { |
|
27
|
0
|
|
|
|
|
|
'inuse' => $inuse, |
|
28
|
|
|
|
|
|
|
'datasize' => $datasize, |
|
29
|
|
|
|
|
|
|
'linkindex' => $linkindex, |
|
30
|
|
|
|
|
|
|
'codename' => $codestr, |
|
31
|
|
|
|
|
|
|
'xor' => $storedxor, |
|
32
|
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
'calcxor' => $calcxor, |
|
34
|
|
|
|
|
|
|
'calcblocks' => $blockcount |
|
35
|
|
|
|
|
|
|
}; |
|
36
|
|
|
|
|
|
|
} |
|
37
|
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
sub parse_file_header { |
|
39
|
0
|
|
|
0
|
0
|
|
my ($file) = @_; |
|
40
|
0
|
|
|
|
|
|
my $id = unpack('a2', $file); |
|
41
|
0
|
|
|
|
|
|
my $displayflag = unpack('C', substr($file, 0x2)); |
|
42
|
0
|
|
|
|
|
|
my $blocknum = unpack('C', substr($file, 0x3)); |
|
43
|
0
|
|
|
|
|
|
my $shiftjisbuf = unpack('a64', substr($file, 0x4, 0x40)); |
|
44
|
0
|
|
|
|
|
|
my @clut = unpack('v16', substr($file, 0x60, 0x20)); |
|
45
|
|
|
|
|
|
|
|
|
46
|
|
|
|
|
|
|
|
|
47
|
0
|
|
|
|
|
|
my $iconfnt = 0; |
|
48
|
0
|
0
|
0
|
|
|
|
if(($displayflag >= 0x11)|| ($displayflag <= 0x13)) { |
|
49
|
0
|
|
|
|
|
|
$iconfnt = $displayflag - 0x10; |
|
50
|
|
|
|
|
|
|
} |
|
51
|
|
|
|
|
|
|
|
|
52
|
0
|
|
|
|
|
|
my $firstnul = index($shiftjisbuf, "\0"); |
|
53
|
0
|
0
|
|
|
|
|
if($firstnul != -1) { |
|
54
|
0
|
|
|
|
|
|
$shiftjisbuf = substr($shiftjisbuf, 0, $firstnul); |
|
55
|
|
|
|
|
|
|
} |
|
56
|
0
|
|
|
|
|
|
my $shiftjis = decode('shiftjis', $shiftjisbuf); |
|
57
|
|
|
|
|
|
|
|
|
58
|
|
|
|
|
|
|
return { |
|
59
|
0
|
|
|
|
|
|
'id' => $id, |
|
60
|
|
|
|
|
|
|
'displayflag' => $displayflag, |
|
61
|
|
|
|
|
|
|
'blocknum' => $blocknum, |
|
62
|
|
|
|
|
|
|
'titlebuf' => $shiftjisbuf, |
|
63
|
|
|
|
|
|
|
'clut' => \@clut, |
|
64
|
|
|
|
|
|
|
'title' => $shiftjis, |
|
65
|
|
|
|
|
|
|
'framecnt' => $iconfnt |
|
66
|
|
|
|
|
|
|
}; |
|
67
|
|
|
|
|
|
|
} |
|
68
|
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
sub is_mcd { |
|
70
|
0
|
|
|
0
|
0
|
|
my ($res) = @_; |
|
71
|
|
|
|
|
|
|
|
|
72
|
0
|
0
|
|
|
|
|
(substr($res, 0, 2) eq 'MC') or return 0; |
|
73
|
|
|
|
|
|
|
|
|
74
|
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
# A PSX memory card is 1 Mebibyte/ 128 kibibyte/ 131072 bytes |
|
76
|
|
|
|
|
|
|
# 1 header block of 8192 and 15 data blocks of 8192. |
|
77
|
0
|
|
|
|
|
|
return (length($res) == 131072); |
|
78
|
|
|
|
|
|
|
} |
|
79
|
|
|
|
|
|
|
|
|
80
|
|
|
|
|
|
|
sub is_mcs { |
|
81
|
0
|
|
|
0
|
0
|
|
my ($res) = @_; |
|
82
|
|
|
|
|
|
|
# A PSX mcs save is 1 directory frame and X data frames |
|
83
|
0
|
|
|
|
|
|
my $datasize = length($res) - 0x80; |
|
84
|
0
|
|
|
|
|
|
return (($datasize % 0x2000) == 0); |
|
85
|
|
|
|
|
|
|
} |
|
86
|
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
sub xordirectory { |
|
88
|
0
|
|
|
0
|
0
|
|
my ($directory) = @_; |
|
89
|
0
|
|
|
|
|
|
my @toxor = unpack('C127', $directory); |
|
90
|
0
|
|
|
|
|
|
my $xor = 0; |
|
91
|
0
|
|
|
|
|
|
foreach my $char (@toxor) { |
|
92
|
0
|
|
|
|
|
|
$xor ^= $char; |
|
93
|
|
|
|
|
|
|
} |
|
94
|
0
|
|
|
|
|
|
return $xor; |
|
95
|
|
|
|
|
|
|
} |
|
96
|
|
|
|
|
|
|
|
|
97
|
|
|
|
|
|
|
|
|
98
|
|
|
|
|
|
|
sub load { |
|
99
|
0
|
|
|
0
|
0
|
|
my ($class, $filename, $overridefilename) = @_; |
|
100
|
|
|
|
|
|
|
|
|
101
|
0
|
|
|
|
|
|
my $fh; |
|
102
|
0
|
0
|
|
|
|
|
if($filename ne '-') { |
|
103
|
0
|
0
|
|
|
|
|
open($fh, '<', $filename) or die("failed to open: $filename"); |
|
104
|
|
|
|
|
|
|
} |
|
105
|
|
|
|
|
|
|
else { |
|
106
|
0
|
|
|
|
|
|
$fh = *STDIN; |
|
107
|
0
|
|
|
|
|
|
$filename = 'STDIN'; |
|
108
|
|
|
|
|
|
|
} |
|
109
|
0
|
|
|
|
|
|
my %self = ('filename' => $filename, 'contents' => ''); |
|
110
|
0
|
|
|
|
|
|
my $res = read($fh, $self{'contents'}, 131073); |
|
111
|
|
|
|
|
|
|
|
|
112
|
|
|
|
|
|
|
# a mcd file (full memory card dump) should be the largest file |
|
113
|
0
|
0
|
0
|
|
|
|
(($res) && ($res <= 131072)) or return undef; |
|
114
|
|
|
|
|
|
|
|
|
115
|
0
|
0
|
0
|
|
|
|
if(is_mcd($self{'contents'})) { |
|
|
|
0
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
116
|
0
|
|
|
|
|
|
$self{'type'} = 'mcd'; |
|
117
|
|
|
|
|
|
|
} |
|
118
|
|
|
|
|
|
|
elsif(is_mcs($self{'contents'})) { |
|
119
|
0
|
|
|
|
|
|
$self{'type'} = 'mcs'; |
|
120
|
|
|
|
|
|
|
} |
|
121
|
|
|
|
|
|
|
elsif(($res <= (15*0x2000)) && ($res >= 0x2000) && (($res % 0x2000) == 0)) { |
|
122
|
0
|
|
|
|
|
|
$self{'type'} = 'rawsave'; |
|
123
|
0
|
0
|
|
|
|
|
my $filecodename = ($filename ne 'STDIN') ? basename($filename) : undef; |
|
124
|
0
|
0
|
|
|
|
|
$self{'codename'} = $overridefilename ? $overridefilename : $filecodename; |
|
125
|
|
|
|
|
|
|
} |
|
126
|
|
|
|
|
|
|
elsif(substr($res, 0, 2) eq 'MC') { |
|
127
|
0
|
|
|
|
|
|
warn("File starts with MC, but filesize is $res. Assuming type is mcd"); |
|
128
|
0
|
|
|
|
|
|
$self{'type'} = 'mcd'; |
|
129
|
|
|
|
|
|
|
} |
|
130
|
|
|
|
|
|
|
else { |
|
131
|
0
|
|
|
|
|
|
return undef; |
|
132
|
|
|
|
|
|
|
} |
|
133
|
|
|
|
|
|
|
|
|
134
|
0
|
0
|
0
|
|
|
|
if(($self{'type'} eq 'mcd') || ($self{'type'} eq 'mcs')) { |
|
135
|
0
|
|
|
|
|
|
$self{'hasdir'} = 1; |
|
136
|
|
|
|
|
|
|
} |
|
137
|
|
|
|
|
|
|
|
|
138
|
0
|
|
|
|
|
|
bless \%self, $class; |
|
139
|
0
|
|
|
|
|
|
return \%self; |
|
140
|
|
|
|
|
|
|
} |
|
141
|
|
|
|
|
|
|
|
|
142
|
|
|
|
|
|
|
# loop through the directory entries of a MCD file, calling callback for each one |
|
143
|
|
|
|
|
|
|
# if it's a startblock read in the save as pass it to the callback |
|
144
|
|
|
|
|
|
|
sub foreachDirEntry { |
|
145
|
0
|
|
|
0
|
0
|
|
my ($self, $callback) = @_; |
|
146
|
0
|
0
|
|
|
|
|
($self->{'type'} eq 'mcd') or die("Unhandled filetype"); |
|
147
|
|
|
|
|
|
|
|
|
148
|
0
|
|
|
|
|
|
my $startindex = 1; |
|
149
|
0
|
|
|
|
|
|
my $dataoffset = 0x2000; |
|
150
|
0
|
|
|
|
|
|
my $maxcount = 15; |
|
151
|
|
|
|
|
|
|
|
|
152
|
0
|
|
|
|
|
|
for(my $i = $startindex; $i < ($startindex+$maxcount); $i++) { |
|
153
|
0
|
|
|
|
|
|
my $entrydata = substr($self->{'contents'}, ($i * 0x80), 0x80); |
|
154
|
0
|
|
|
|
|
|
my $entry = parse_directory($entrydata); |
|
155
|
0
|
|
|
|
|
|
my $save; |
|
156
|
0
|
0
|
|
|
|
|
if($entry->{'inuse'} == 0x51) { |
|
157
|
|
|
|
|
|
|
$save = { |
|
158
|
|
|
|
|
|
|
'filename' => $entry->{'codename'}, |
|
159
|
0
|
|
|
|
|
|
'data' => substr($self->{'contents'}, $dataoffset, $entry->{'datasize'}), |
|
160
|
|
|
|
|
|
|
}; |
|
161
|
|
|
|
|
|
|
} |
|
162
|
0
|
|
|
|
|
|
$callback->($entry, $save, $entrydata); |
|
163
|
0
|
|
|
|
|
|
$dataoffset += 0x2000; |
|
164
|
|
|
|
|
|
|
} |
|
165
|
|
|
|
|
|
|
} |
|
166
|
|
|
|
|
|
|
|
|
167
|
|
|
|
|
|
|
sub _readMCSSave { |
|
168
|
0
|
|
|
0
|
|
|
my ($self) = @_; |
|
169
|
0
|
|
|
|
|
|
my $entrydata = substr($self->{'contents'}, 0, 0x80); |
|
170
|
0
|
|
|
|
|
|
my $entry = parse_directory($entrydata); |
|
171
|
|
|
|
|
|
|
|
|
172
|
|
|
|
|
|
|
return { |
|
173
|
|
|
|
|
|
|
'filename' => $entry->{'codename'}, |
|
174
|
0
|
|
|
|
|
|
'data' => substr($self->{'contents'}, 0x80, $entry->{'datasize'}) |
|
175
|
|
|
|
|
|
|
}; |
|
176
|
|
|
|
|
|
|
} |
|
177
|
|
|
|
|
|
|
|
|
178
|
|
|
|
|
|
|
sub _readRawSave { |
|
179
|
0
|
|
|
0
|
|
|
my ($self) = @_; |
|
180
|
0
|
0
|
|
|
|
|
$self->{'codename'} or die("cannot read raw save without a filename"); |
|
181
|
|
|
|
|
|
|
return { |
|
182
|
|
|
|
|
|
|
'filename' => $self->{'codename'}, |
|
183
|
0
|
|
|
|
|
|
'data' => $self->{'contents'} |
|
184
|
|
|
|
|
|
|
}; |
|
185
|
|
|
|
|
|
|
} |
|
186
|
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
|
|
188
|
|
|
|
|
|
|
sub readSave { |
|
189
|
0
|
|
|
0
|
0
|
|
my ($self) = @_; |
|
190
|
0
|
0
|
|
|
|
|
if($self->{'type'} eq 'mcs') { |
|
|
|
0
|
|
|
|
|
|
|
191
|
0
|
|
|
|
|
|
return _readMCSSave($self); |
|
192
|
|
|
|
|
|
|
} |
|
193
|
|
|
|
|
|
|
elsif($self->{'type'} eq 'rawsave') { |
|
194
|
0
|
|
|
|
|
|
return _readRawSave($self); |
|
195
|
|
|
|
|
|
|
} |
|
196
|
|
|
|
|
|
|
else { |
|
197
|
0
|
|
|
|
|
|
die("unimplemented type"); |
|
198
|
|
|
|
|
|
|
} |
|
199
|
|
|
|
|
|
|
} |
|
200
|
|
|
|
|
|
|
|
|
201
|
|
|
|
|
|
|
sub FormatSaveFirstDirEntry { |
|
202
|
0
|
|
|
0
|
0
|
|
my ($save, $dirindex) = @_; |
|
203
|
0
|
|
|
|
|
|
my $savelen = length($save->{'data'}); |
|
204
|
0
|
|
|
|
|
|
my $blockcount = length($save->{'data'}) / 0x2000; |
|
205
|
0
|
0
|
|
|
|
|
(($blockcount % 1) == 0) or die("not integer blocksize"); |
|
206
|
0
|
0
|
|
|
|
|
($blockcount >= 1) or die("must have at least one block"); |
|
207
|
0
|
0
|
|
|
|
|
my $blockptr = ($blockcount == 1) ? 0xFFFF : $dirindex; |
|
208
|
0
|
|
|
|
|
|
my $directory = pack('VVvZ21x96', 0x51, $savelen, $blockptr, $save->{'filename'}); |
|
209
|
0
|
|
|
|
|
|
$directory .= pack('C', xordirectory($directory)); |
|
210
|
|
|
|
|
|
|
|
|
211
|
0
|
|
|
|
|
|
return ($directory, $blockcount); |
|
212
|
|
|
|
|
|
|
} |
|
213
|
|
|
|
|
|
|
|
|
214
|
|
|
|
|
|
|
sub FormatSaveAsMCD { |
|
215
|
0
|
|
|
0
|
0
|
|
my ($dirstart, $save) = @_; |
|
216
|
|
|
|
|
|
|
|
|
217
|
0
|
|
|
|
|
|
my $dirindex = ($dirstart / 0x80); |
|
218
|
0
|
|
|
|
|
|
my ($directory, $blockcount) = FormatSaveFirstDirEntry($save, $dirindex); |
|
219
|
|
|
|
|
|
|
|
|
220
|
|
|
|
|
|
|
# format the possible mid and end link directories |
|
221
|
0
|
0
|
|
|
|
|
if($blockcount > 1) { |
|
222
|
0
|
|
|
|
|
|
while ($blockcount > 2) { |
|
223
|
0
|
|
|
|
|
|
$dirindex++; |
|
224
|
0
|
|
|
|
|
|
my $newdir = pack('VVvx117', 0x52, 0x0, $dirindex); |
|
225
|
0
|
|
|
|
|
|
$newdir .= pack('C', xordirectory($newdir)); |
|
226
|
0
|
|
|
|
|
|
$directory .= $newdir; |
|
227
|
0
|
|
|
|
|
|
$blockcount--; |
|
228
|
|
|
|
|
|
|
} |
|
229
|
0
|
|
|
|
|
|
my $newdir = pack('VVvx117', 0x53, 0x0, 0xFFFF); |
|
230
|
0
|
|
|
|
|
|
$newdir .= pack('C', xordirectory($newdir)); |
|
231
|
0
|
|
|
|
|
|
$directory .= $newdir; |
|
232
|
|
|
|
|
|
|
} |
|
233
|
|
|
|
|
|
|
|
|
234
|
|
|
|
|
|
|
return { |
|
235
|
|
|
|
|
|
|
'dirdata' => $directory, |
|
236
|
0
|
|
|
|
|
|
'savedata' => $save->{'data'} |
|
237
|
|
|
|
|
|
|
}; |
|
238
|
|
|
|
|
|
|
} |
|
239
|
|
|
|
|
|
|
|
|
240
|
|
|
|
|
|
|
sub FormatSaveAsMCS { |
|
241
|
0
|
|
|
0
|
0
|
|
my ($save) = @_; |
|
242
|
0
|
|
|
|
|
|
my ($directory, $blockcount) = FormatSaveFirstDirEntry($save, 1); |
|
243
|
0
|
|
|
|
|
|
return $directory .= $save->{'data'}; |
|
244
|
|
|
|
|
|
|
} |
|
245
|
|
|
|
|
|
|
|
|
246
|
|
|
|
|
|
|
sub SaveNameAndTitleMatch { |
|
247
|
0
|
|
|
0
|
0
|
|
my ($save, $string) = @_; |
|
248
|
0
|
|
0
|
|
|
|
$save->{'header'} //= PlayStation::MemoryCard::parse_file_header($save->{'data'}); |
|
249
|
0
|
|
|
|
|
|
my $title = $save->{'header'}{'title'}; |
|
250
|
0
|
|
|
|
|
|
my $asciititle = $title; |
|
251
|
1
|
|
|
1
|
|
604
|
$asciititle =~ tr/\x{3000}\x{FF01}-\x{FF5E}/ -~/; # fullwidth to half-width if possible |
|
|
1
|
|
|
|
|
15
|
|
|
|
1
|
|
|
|
|
15
|
|
|
|
0
|
|
|
|
|
|
|
|
252
|
|
|
|
|
|
|
#warn("title: $title"); |
|
253
|
|
|
|
|
|
|
#warn("searchfname: $string"); |
|
254
|
|
|
|
|
|
|
#warn("asciititle: $asciititle"); |
|
255
|
|
|
|
|
|
|
return (!$string || |
|
256
|
|
|
|
|
|
|
($save->{'filename'} eq $string) || ($title eq $string) || ($asciititle eq $string) || |
|
257
|
0
|
|
0
|
|
|
|
($save->{'filename'} =~ /\Q$string\E/i) || ($title =~ /\Q$string\E/i) || ($asciititle =~ /\Q$string\E/i)); |
|
258
|
|
|
|
|
|
|
} |
|
259
|
|
|
|
|
|
|
|
|
260
|
|
|
|
|
|
|
sub BlankMCD { |
|
261
|
0
|
|
|
0
|
0
|
|
my $cardbuf = pack('x131072'); |
|
262
|
|
|
|
|
|
|
|
|
263
|
|
|
|
|
|
|
# header frame |
|
264
|
0
|
|
|
|
|
|
substr($cardbuf, 0, 2, 'MC'); |
|
265
|
0
|
|
|
|
|
|
substr($cardbuf, 0x7F, 1, pack('C', 0x0E)); |
|
266
|
|
|
|
|
|
|
# directory frames |
|
267
|
0
|
|
|
|
|
|
for(my $i = 1; $i < 16; $i++) { |
|
268
|
0
|
|
|
|
|
|
my $frameoffset = $i*0x80; |
|
269
|
0
|
|
|
|
|
|
substr($cardbuf, $frameoffset, 1, pack('C', 0xA0)); |
|
270
|
0
|
|
|
|
|
|
substr($cardbuf, $frameoffset+0x8, 2, pack('v', 0xFFFF)); |
|
271
|
0
|
|
|
|
|
|
substr($cardbuf, $frameoffset+0x7F, 1, pack('C', 0xA0)); |
|
272
|
|
|
|
|
|
|
} |
|
273
|
|
|
|
|
|
|
# broken sector list |
|
274
|
0
|
|
|
|
|
|
for(my $i = 16; $i < 36; $i++) { |
|
275
|
0
|
|
|
|
|
|
my $frameoffset = $i*0x80; |
|
276
|
0
|
|
|
|
|
|
substr($cardbuf, $frameoffset, 4, pack('V', 0xFFFFFFFF)); |
|
277
|
0
|
|
|
|
|
|
substr($cardbuf, $frameoffset+0x8, 2, pack('v', 0xFFFF)); |
|
278
|
|
|
|
|
|
|
} |
|
279
|
|
|
|
|
|
|
# broken sector replacement data 36-55 |
|
280
|
|
|
|
|
|
|
# unused frames 56-62 |
|
281
|
|
|
|
|
|
|
# write test frame 63 |
|
282
|
|
|
|
|
|
|
# file blocks |
|
283
|
|
|
|
|
|
|
|
|
284
|
0
|
|
|
|
|
|
return $cardbuf; |
|
285
|
|
|
|
|
|
|
} |
|
286
|
|
|
|
|
|
|
|
|
287
|
|
|
|
|
|
|
1; |
|
288
|
|
|
|
|
|
|
|
|
289
|
|
|
|
|
|
|
__END__ |