1 | #include <string.h>
|
---|
2 |
|
---|
3 | #include "Daodan.h"
|
---|
4 | #include "Daodan_Patch.h"
|
---|
5 | #include "Daodan_Utility.h"
|
---|
6 | #include "Daodan_Win32.h"
|
---|
7 | #include "Daodan_Cheater.h"
|
---|
8 | #include "Daodan_Persistence.h"
|
---|
9 |
|
---|
10 | #include "Daodan_WindowHack.h"
|
---|
11 |
|
---|
12 | #include "Oni.h"
|
---|
13 | #include "Oni_Persistence.h"
|
---|
14 |
|
---|
15 | #include "BFW_Utility.h"
|
---|
16 |
|
---|
17 | #include "oni_gl.h"
|
---|
18 | #include "daodan_gl.h"
|
---|
19 |
|
---|
20 | #include "inifile.h"
|
---|
21 |
|
---|
22 | HMODULE DDrDLLModule;
|
---|
23 | HMODULE DDrONiModule;
|
---|
24 |
|
---|
25 | bool patch_fonttexturecache = true;
|
---|
26 | bool patch_largetextures = true;
|
---|
27 | bool patch_levelplugins = true;
|
---|
28 | bool patch_pathfinding = true;
|
---|
29 | bool patch_projaware = true;
|
---|
30 | bool patch_directinput = true;
|
---|
31 | bool patch_wpfadetime = true;
|
---|
32 | bool patch_kickguns = false;
|
---|
33 | bool patch_cooldowntimer = true;
|
---|
34 | bool patch_throwtest = false;
|
---|
35 | bool patch_alttab = true;
|
---|
36 | bool patch_particledisablebit = false;
|
---|
37 | bool patch_multibyte = false;
|
---|
38 | bool patch_cheattable = true;
|
---|
39 |
|
---|
40 | bool patch_safeprintf = true;
|
---|
41 | bool patch_daodandisplayenum = true;
|
---|
42 | bool patch_usegettickcount = true;
|
---|
43 | bool patch_cheatsenabled = true;
|
---|
44 | bool patch_usedaodangl = false;
|
---|
45 | bool patch_windowhack = true;
|
---|
46 |
|
---|
47 | bool DDrPatch_Init()
|
---|
48 | {
|
---|
49 | DDrStartupMessage("patching engine");
|
---|
50 |
|
---|
51 | // Font texture cache doubled
|
---|
52 | if (patch_fonttexturecache)
|
---|
53 | {
|
---|
54 | DDrPatch_Byte (OniExe + 0x00020ea7, 0x20);
|
---|
55 | DDrPatch_Byte (OniExe + 0x00020f4a, 0x40);
|
---|
56 | }
|
---|
57 |
|
---|
58 | // Now supports textures up to 512x512
|
---|
59 | if (patch_largetextures)
|
---|
60 | DDrPatch_Byte (OniExe + 0x00005251, 0x10);
|
---|
61 |
|
---|
62 | // Non-"_Final" levels are now valid
|
---|
63 | if (patch_levelplugins)
|
---|
64 | DDrPatch_Byte (OniExe + 0x000206a8, 0x01);
|
---|
65 |
|
---|
66 | // Pathfinding grid cache size x8
|
---|
67 | if (patch_pathfinding)
|
---|
68 | {
|
---|
69 | DDrPatch_Byte (OniExe + 0x0010b03b, 0x20);
|
---|
70 | DDrPatch_Byte (OniExe + 0x0010b04c, 0x20);
|
---|
71 | }
|
---|
72 |
|
---|
73 | // Projectile awareness fixed
|
---|
74 | if (patch_projaware)
|
---|
75 | {
|
---|
76 | DDrPatch_Byte (OniExe + 0x0009c07c, 0x6c);
|
---|
77 | DDrPatch_Byte (OniExe + 0x0009c080, 0x70);
|
---|
78 | DDrPatch_Byte (OniExe + 0x0009c084, 0x74);
|
---|
79 | DDrPatch_Byte (OniExe + 0x0009c110, 0x6c);
|
---|
80 | }
|
---|
81 |
|
---|
82 | // Forced DirectInput (for Windows NT)
|
---|
83 | if (patch_directinput)
|
---|
84 | DDrPatch_Byte (OniExe + 0x00002e6d, 0xeb);
|
---|
85 |
|
---|
86 | if (patch_wpfadetime)
|
---|
87 | {
|
---|
88 | // Makes wp_fadetime actually have a function
|
---|
89 | const char fadetime_patch[] = { 0x66, 0x8B, 0x1D, 0xC4, 0x7D, 0x62, 0x00, 0x66, 0x89, 0x5E, 0x46, 0x5B, 0x5E, 0x83, 0xC4, 0x14, 0xC3 };
|
---|
90 | DDrPatch_Const (OniExe + 0x0011a889, fadetime_patch);
|
---|
91 | DDrPatch_Byte (OniExe + 0x0011a560, 0x31);
|
---|
92 |
|
---|
93 | // Sets the fadetime to 4800 by default
|
---|
94 | DDrPatch_Int16 (OniExe + 0x0011ab0e, 0x12c0);
|
---|
95 | }
|
---|
96 |
|
---|
97 |
|
---|
98 | // Hackish fix for Konoko not kicking guns
|
---|
99 | if (patch_kickguns)
|
---|
100 | {
|
---|
101 | const char kickgun_patch[] = { 0x00, 0x05, 0x00, 0x00, 0x00, 0xC7, 0x05, 0x1C, 0xC9, 0x5E, 0x00, 0x70, 0xB8, 0x43, 0x00, 0xC7, 0x05, 0x20, 0xC9, 0x5E, 0x00, 0x20, 0xBE, 0x43 };
|
---|
102 | DDrPatch_Const (OniExe + 0x000dc420, kickgun_patch);
|
---|
103 | }
|
---|
104 |
|
---|
105 | // Cooldown timer exploit fix ^_^
|
---|
106 | if (patch_cooldowntimer)
|
---|
107 | {
|
---|
108 | const char cooldown_patch[] = { 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 };
|
---|
109 | DDrPatch_Const (OniExe + 0x0011a825, cooldown_patch);
|
---|
110 | }
|
---|
111 |
|
---|
112 | if (patch_throwtest)
|
---|
113 | {
|
---|
114 | const char throwtest_patch[] = { 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 };
|
---|
115 | DDrPatch_Const(OniExe + 0x000dc190, throwtest_patch);
|
---|
116 | }
|
---|
117 |
|
---|
118 | // Disable UUrPlatform_Initalize/Terminate, this enables the Alt-Tab and the Windows key but has the possible side effect of allowing the screensaver to enable itself in-game.
|
---|
119 | if (patch_alttab)
|
---|
120 | {
|
---|
121 | DDrPatch_Byte ((void*)UUrPlatform_Initialize, 0xC3);
|
---|
122 | DDrPatch_Byte ((void*)UUrPlatform_Terminate, 0xC3);
|
---|
123 | }
|
---|
124 |
|
---|
125 | // Unlocks particle action disabling/enabling bits for all events. (Will be controlled by a command line switch when I figure out how to do that without Win32 hacks.)
|
---|
126 | if (patch_particledisablebit)
|
---|
127 | DDrPatch_Int16 (OniExe + 0x001b184, 0x9090);
|
---|
128 |
|
---|
129 | // Multi-byte patch (multiple language support)
|
---|
130 | if (!patch_multibyte)
|
---|
131 | {
|
---|
132 | DDrPatch_Byte (OniExe + 0x0002d8f8, 0xeb);
|
---|
133 | DDrPatch_Byte (OniExe + 0x0002d9ad, 0xeb);
|
---|
134 | DDrPatch_Byte (OniExe + 0x0002dbe2, 0xeb);
|
---|
135 | DDrPatch_Byte (OniExe + 0x0002dec3, 0xeb);
|
---|
136 | DDrPatch_Byte (OniExe + 0x0002e2ab, 0xeb);
|
---|
137 | DDrPatch_Byte (OniExe + 0x0002e2c4, 0xeb);
|
---|
138 | DDrPatch_Byte (OniExe + 0x0002e379, 0xeb);
|
---|
139 | DDrPatch_Byte (OniExe + 0x0002e48c, 0xeb);
|
---|
140 | DDrPatch_Byte (OniExe + 0x0002e4d0, 0xeb);
|
---|
141 | DDrPatch_Byte (OniExe + 0x0002e4f4, 0xeb);
|
---|
142 | DDrPatch_Byte (OniExe + 0x0002e646, 0xeb);
|
---|
143 | DDrPatch_Byte (OniExe + 0x0002e695, 0xeb);
|
---|
144 | DDrPatch_Byte (OniExe + 0x0002e944, 0xeb);
|
---|
145 | DDrPatch_Byte (OniExe + 0x0002e95d, 0xeb);
|
---|
146 | DDrPatch_Byte (OniExe + 0x0002e98e, 0xeb);
|
---|
147 | DDrPatch_Byte (OniExe + 0x0002e9dc, 0xeb);
|
---|
148 | }
|
---|
149 |
|
---|
150 | // Cheat table patch
|
---|
151 | if (patch_cheattable)
|
---|
152 | {
|
---|
153 | DDrPatch_Int32 (OniExe + 0x000f616b, (int)&DDr_CheatTable[0].name);
|
---|
154 | DDrPatch_Int32 (OniExe + 0x000f617a, (int)&DDr_CheatTable[0].message_on);
|
---|
155 | }
|
---|
156 |
|
---|
157 | return true;
|
---|
158 | }
|
---|
159 |
|
---|
160 | enum {s_unknown, s_patch, s_language} ini_section;
|
---|
161 |
|
---|
162 | bool DDrIniCallback(char* section, bool newsection, char* name, char* value)
|
---|
163 | {
|
---|
164 | if (newsection)
|
---|
165 | {
|
---|
166 | if (!stricmp(section, "patch"))
|
---|
167 | ini_section = s_patch;
|
---|
168 | else if (!stricmp(section, "language"))
|
---|
169 | ini_section = s_language;
|
---|
170 | else
|
---|
171 | {
|
---|
172 | ini_section = s_unknown;
|
---|
173 | DDrStartupMessage("unrecognised ini section \"%s\"", section);
|
---|
174 | }
|
---|
175 | }
|
---|
176 |
|
---|
177 | switch (ini_section)
|
---|
178 | {
|
---|
179 | case s_patch:
|
---|
180 | if (!stricmp(name, "fonttexturecache"))
|
---|
181 | patch_fonttexturecache = !stricmp(value, "true");
|
---|
182 | else if (!stricmp(name, "largetextures"))
|
---|
183 | patch_largetextures = !stricmp(value, "true");
|
---|
184 | else if (!stricmp(name, "levelplugins"))
|
---|
185 | patch_levelplugins = !stricmp(value, "true");
|
---|
186 | else if (!stricmp(name, "pathfinding"))
|
---|
187 | patch_pathfinding = !stricmp(value, "true");
|
---|
188 | else if (!stricmp(name, "projaware"))
|
---|
189 | patch_projaware = !stricmp(value, "true");
|
---|
190 | else if (!stricmp(name, "directinput"))
|
---|
191 | patch_directinput = !stricmp(value, "true");
|
---|
192 | else if (!stricmp(name, "wpfadetime"))
|
---|
193 | patch_wpfadetime = !stricmp(value, "true");
|
---|
194 | else if (!stricmp(name, "kickguns"))
|
---|
195 | patch_kickguns = !stricmp(value, "true");
|
---|
196 | else if (!stricmp(name, "cooldowntimer"))
|
---|
197 | patch_cooldowntimer = !stricmp(value, "true");
|
---|
198 | else if (!stricmp(name, "throwtest"))
|
---|
199 | patch_throwtest = !stricmp(value, "true");
|
---|
200 | else if (!stricmp(name, "alttab"))
|
---|
201 | patch_alttab = !stricmp(value, "true");
|
---|
202 | else if (!stricmp(name, "particledisablebit"))
|
---|
203 | patch_particledisablebit = !stricmp(value, "true");
|
---|
204 | else if (!stricmp(name, "multibyte"))
|
---|
205 | patch_multibyte = !stricmp(value, "true");
|
---|
206 | else if (!stricmp(name, "cheattable"))
|
---|
207 | patch_cheattable = !stricmp(value, "true");
|
---|
208 | else if (!stricmp(name, "safeprintf"))
|
---|
209 | patch_safeprintf = !stricmp(value, "true");
|
---|
210 | else if (!stricmp(name, "daodandisplayenum"))
|
---|
211 | patch_daodandisplayenum = !stricmp(value, "true");
|
---|
212 | else if (!stricmp(name, "usegettickcount"))
|
---|
213 | patch_usegettickcount = !stricmp(value, "true");
|
---|
214 | else if (!stricmp(name, "cheatsenabled"))
|
---|
215 | patch_cheatsenabled = !stricmp(value, "true");
|
---|
216 | else if (!stricmp(name, "usedaodangl"))
|
---|
217 | patch_usedaodangl = !stricmp(value, "true");
|
---|
218 | else if (!stricmp(name, "windowhack"))
|
---|
219 | patch_windowhack = !stricmp(value, "true");
|
---|
220 | else
|
---|
221 | DDrStartupMessage("unrecognised patch \"%s\"", name);
|
---|
222 | break;
|
---|
223 | case s_language:
|
---|
224 | if (!stricmp(name, "savepoint"))
|
---|
225 | {
|
---|
226 | char* str = strdup(value);
|
---|
227 | DDrPatch_Int32(OniExe + 0x000fd730, (int)str);
|
---|
228 | DDrPatch_Int32(OniExe + 0x000fd738, (int)str);
|
---|
229 | }
|
---|
230 | else if (!stricmp(name, "syndicatewarehouse"))
|
---|
231 | {
|
---|
232 | char* str = strdup(value);
|
---|
233 | DDrPatch_Int32(OniExe + 0x000fd71a, (int)str);
|
---|
234 | DDrPatch_Int32(OniExe + 0x0010ef75, (int)str);
|
---|
235 | }
|
---|
236 | else if (!stricmp(name, "damn"))
|
---|
237 | DDrPatch_StrDup(OniExe + 0x0010fb6e, value);
|
---|
238 | else if (!stricmp(name, "blam"))
|
---|
239 | DDrPatch_StrDup(OniExe + 0x0010fb73, value);
|
---|
240 | else if (!stricmp(name, "shapeshifter_on"))
|
---|
241 | DDr_CheatTable[0].message_on = strdup(value);
|
---|
242 | else if (!stricmp(name, "shapeshifter_off"))
|
---|
243 | DDr_CheatTable[0].message_off = strdup(value);
|
---|
244 | else if (!stricmp(name, "liveforever_on"))
|
---|
245 | DDr_CheatTable[1].message_on = strdup(value);
|
---|
246 | else if (!stricmp(name, "liveforever_off"))
|
---|
247 | DDr_CheatTable[1].message_off = strdup(value);
|
---|
248 | else if (!stricmp(name, "touchofdeath_on"))
|
---|
249 | DDr_CheatTable[2].message_on = strdup(value);
|
---|
250 | else if (!stricmp(name, "touchofdeath_off"))
|
---|
251 | DDr_CheatTable[2].message_off = strdup(value);
|
---|
252 | else if (!stricmp(name, "canttouchthis_on"))
|
---|
253 | DDr_CheatTable[3].message_on = strdup(value);
|
---|
254 | else if (!stricmp(name, "canttouchthis_off"))
|
---|
255 | DDr_CheatTable[3].message_off = strdup(value);
|
---|
256 | else if (!stricmp(name, "fatloot_on"))
|
---|
257 | DDr_CheatTable[4].message_on = strdup(value);
|
---|
258 | else if (!stricmp(name, "glassworld_on"))
|
---|
259 | DDr_CheatTable[5].message_on = strdup(value);
|
---|
260 | else if (!stricmp(name, "glassworld_off"))
|
---|
261 | DDr_CheatTable[5].message_off = strdup(value);
|
---|
262 | else if (!stricmp(name, "winlevel_on"))
|
---|
263 | DDr_CheatTable[6].message_on = strdup(value);
|
---|
264 | else if (!stricmp(name, "loselevel_on"))
|
---|
265 | DDr_CheatTable[7].message_on = strdup(value);
|
---|
266 | else if (!stricmp(name, "bighead_on"))
|
---|
267 | DDr_CheatTable[8].message_on = strdup(value);
|
---|
268 | else if (!stricmp(name, "bighead_off"))
|
---|
269 | DDr_CheatTable[8].message_off = strdup(value);
|
---|
270 | else if (!stricmp(name, "minime_on"))
|
---|
271 | DDr_CheatTable[9].message_on = strdup(value);
|
---|
272 | else if (!stricmp(name, "minime_off"))
|
---|
273 | DDr_CheatTable[9].message_off = strdup(value);
|
---|
274 | else if (!stricmp(name, "superammo_on"))
|
---|
275 | DDr_CheatTable[10].message_on = strdup(value);
|
---|
276 | else if (!stricmp(name, "superammo_off"))
|
---|
277 | DDr_CheatTable[10].message_off = strdup(value);
|
---|
278 | else if (!stricmp(name, "devmode_on"))
|
---|
279 | {
|
---|
280 | char* str = strdup(value);
|
---|
281 | DDr_CheatTable[11].message_on = str;
|
---|
282 | DDr_CheatTable[cheat_devmodex].message_on = str;
|
---|
283 | }
|
---|
284 | else if (!stricmp(name, "devmode_off"))
|
---|
285 | {
|
---|
286 | char* str = strdup(value);
|
---|
287 | DDr_CheatTable[11].message_off = str;
|
---|
288 | DDr_CheatTable[cheat_devmodex].message_off = str;
|
---|
289 | }
|
---|
290 | else if (!stricmp(name, "reservoirdogs_on"))
|
---|
291 | DDr_CheatTable[12].message_on = strdup(value);
|
---|
292 | else if (!stricmp(name, "reservoirdogs_off"))
|
---|
293 | DDr_CheatTable[12].message_off = strdup(value);
|
---|
294 | else if (!stricmp(name, "roughjustice_on"))
|
---|
295 | DDr_CheatTable[13].message_on = strdup(value);
|
---|
296 | else if (!stricmp(name, "roughjustice_off"))
|
---|
297 | DDr_CheatTable[13].message_off = strdup(value);
|
---|
298 | else if (!stricmp(name, "chenille_on"))
|
---|
299 | DDr_CheatTable[14].message_on = strdup(value);
|
---|
300 | else if (!stricmp(name, "chenille_off"))
|
---|
301 | DDr_CheatTable[14].message_off = strdup(value);
|
---|
302 | else if (!stricmp(name, "behemoth_on"))
|
---|
303 | DDr_CheatTable[15].message_on = strdup(value);
|
---|
304 | else if (!stricmp(name, "behemoth_off"))
|
---|
305 | DDr_CheatTable[15].message_off = strdup(value);
|
---|
306 | else if (!stricmp(name, "elderrune_on"))
|
---|
307 | DDr_CheatTable[16].message_on = strdup(value);
|
---|
308 | else if (!stricmp(name, "elderrune_off"))
|
---|
309 | DDr_CheatTable[16].message_off = strdup(value);
|
---|
310 | else if (!stricmp(name, "moonshadow_on"))
|
---|
311 | DDr_CheatTable[17].message_on = strdup(value);
|
---|
312 | else if (!stricmp(name, "moonshadow_off"))
|
---|
313 | DDr_CheatTable[17].message_off = strdup(value);
|
---|
314 | else if (!stricmp(name, "munitionfrenzy_on"))
|
---|
315 | DDr_CheatTable[18].message_on = strdup(value);
|
---|
316 | else if (!stricmp(name, "fistsoflegend_on"))
|
---|
317 | DDr_CheatTable[19].message_on = strdup(value);
|
---|
318 | else if (!stricmp(name, "fistsoflegend_off"))
|
---|
319 | DDr_CheatTable[19].message_off = strdup(value);
|
---|
320 | else if (!stricmp(name, "killmequick_on"))
|
---|
321 | DDr_CheatTable[20].message_on = strdup(value);
|
---|
322 | else if (!stricmp(name, "killmequick_off"))
|
---|
323 | DDr_CheatTable[20].message_off = strdup(value);
|
---|
324 | else if (!stricmp(name, "carousel_on"))
|
---|
325 | DDr_CheatTable[21].message_on = strdup(value);
|
---|
326 | else if (!stricmp(name, "carousel_off"))
|
---|
327 | DDr_CheatTable[21].message_off = strdup(value);
|
---|
328 | else
|
---|
329 | DDrStartupMessage("unrecognised language item \"%s\"", name);
|
---|
330 | break;
|
---|
331 | default:
|
---|
332 | break;
|
---|
333 | }
|
---|
334 |
|
---|
335 | return true;
|
---|
336 | }
|
---|
337 |
|
---|
338 | void DDrConfig()
|
---|
339 | {
|
---|
340 | if (GetFileAttributes("daodan.ini") == INVALID_FILE_ATTRIBUTES)
|
---|
341 | {
|
---|
342 | DDrStartupMessage("daodan.ini doesn't exist, creating");
|
---|
343 | FILE* fp = fopen("daodan.ini", "w");
|
---|
344 | if (fp)
|
---|
345 | {
|
---|
346 | fputs("[Options]\n", fp);
|
---|
347 | fclose(fp);
|
---|
348 | }
|
---|
349 | }
|
---|
350 |
|
---|
351 | DDrStartupMessage("parsing daodan.ini...");
|
---|
352 | if (!inifile_read("daodan.ini", DDrIniCallback))
|
---|
353 | DDrStartupMessage("error reading daodan.ini, check your syntax!");
|
---|
354 | DDrStartupMessage("finished parsing");
|
---|
355 | }
|
---|
356 |
|
---|
357 | void __cdecl DDrMain(int argc, char* argv[])
|
---|
358 | {
|
---|
359 | DDrStartupMessage("daodan attached!");
|
---|
360 | DDrConfig();
|
---|
361 | DDrPatch_Init();
|
---|
362 |
|
---|
363 | // Safe startup message printer
|
---|
364 | if (patch_safeprintf)
|
---|
365 | DDrPatch_MakeJump(UUrStartupMessage, DDrStartupMessage);
|
---|
366 |
|
---|
367 | // Daodan device mode enumeration function
|
---|
368 | if (patch_daodandisplayenum)
|
---|
369 | DDrPatch_MakeJump(gl_enumerate_valid_display_modes, daodan_enumerate_valid_display_modes);
|
---|
370 |
|
---|
371 | // Performance patch
|
---|
372 | if (patch_usegettickcount)
|
---|
373 | {
|
---|
374 | DDrPatch_MakeJump(UUrMachineTime_High, DDrMachineTime_High);
|
---|
375 | DDrPatch_MakeJump(UUrMachineTime_High_Frequency, DDrMachineTime_High_Frequency);
|
---|
376 | DDrPatch_MakeJump(UUrMachineTime_Sixtieths, DDrMachineTime_Sixtieths);
|
---|
377 | }
|
---|
378 |
|
---|
379 | // Cheats always enabled
|
---|
380 | if (patch_cheatsenabled)
|
---|
381 | DDrPatch_MakeJump(ONrPersist_GetWonGame, DDrPersist_GetWonGame);
|
---|
382 |
|
---|
383 | // Windowed mode
|
---|
384 | if (patch_usedaodangl)
|
---|
385 | {
|
---|
386 | DDrPatch_MakeJump(ONrPlatform_Initialize, DDrPlatform_Initialize);
|
---|
387 | DDrPatch_MakeJump(gl_platform_initialize, daodangl_platform_initialize);
|
---|
388 | }
|
---|
389 |
|
---|
390 | // Hacked windowed mode (for when daodangl isn't working properly)
|
---|
391 | if (patch_windowhack)
|
---|
392 | DDrWindowHack_Install();
|
---|
393 |
|
---|
394 | init_daodan_gl();
|
---|
395 |
|
---|
396 | ONiMain(argc, argv);
|
---|
397 | }
|
---|
398 |
|
---|
399 | BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpReserved)
|
---|
400 | {
|
---|
401 | switch (fdwReason)
|
---|
402 | {
|
---|
403 | case DLL_PROCESS_ATTACH:
|
---|
404 | DDrDLLModule = hinstDLL;
|
---|
405 | DDrONiModule = GetModuleHandle(NULL);
|
---|
406 |
|
---|
407 | DDrPatch_MakeCall(OniExe + 0x0010fb49, DDrMain);
|
---|
408 |
|
---|
409 | break;
|
---|
410 | }
|
---|
411 | return TRUE;
|
---|
412 | }
|
---|