1 /*  RetroArch - A frontend for libretro.
2  *  Copyright (C) 2010-2014 - Hans-Kristian Arntzen
3  *  Copyright (C) 2011-2017 - Daniel De Matteis
4  *
5  *  RetroArch is free software: you can redistribute it and/or modify it under the terms
6  *  of the GNU General Public License as published by the Free Software Found-
7  *  ation, either version 3 of the License, or (at your option) any later version.
8  *
9  *  RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
10  *  without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
11  *  PURPOSE.  See the GNU General Public License for more details.
12  *
13  *  You should have received a copy of the GNU General Public License along with RetroArch.
14  *  If not, see <http://www.gnu.org/licenses/>.
15  */
16 
17 #include <stdint.h>
18 #include <stddef.h>
19 
20 #include <xtl.h>
21 #include <xbdm.h>
22 #include <xgraphics.h>
23 
24 #include <file/file_path.h>
25 #include <compat/strl.h>
26 
27 #ifdef HAVE_CONFIG_H
28 #include "../../config.h"
29 #endif
30 
31 #ifndef IS_SALAMANDER
32 #include <lists/file_list.h>
33 #endif
34 #include <retro_miscellaneous.h>
35 #include <string/stdstring.h>
36 
37 #include "../frontend_driver.h"
38 #include "../../defaults.h"
39 #include "../../file_path_special.h"
40 #include "../../paths.h"
41 #ifndef IS_SALAMANDER
42 #include "../../retroarch.h"
43 #ifdef HAVE_MENU
44 #include "../../menu/menu_driver.h"
45 #endif
46 #endif
47 #include "../../verbosity.h"
48 
49 #include "platform_xdk.h"
50 
51 static enum frontend_fork xdk_fork_mode = FRONTEND_FORK_NONE;
52 
frontend_xdk_get_environment_settings(int * argc,char * argv[],void * args,void * params_data)53 static void frontend_xdk_get_environment_settings(int *argc, char *argv[],
54       void *args, void *params_data)
55 {
56    HRESULT ret;
57 #ifdef _XBOX360
58    unsigned long license_mask;
59    DWORD volume_device_type;
60 #endif
61 #ifndef IS_SALAMANDER
62    static char path[PATH_MAX_LENGTH] = {0};
63 #if defined(_XBOX1)
64    LAUNCH_DATA ptr;
65    DWORD launch_type;
66 #elif defined(_XBOX360)
67    DWORD dwLaunchDataSize;
68 #endif
69 #endif
70 #ifndef IS_SALAMANDER
71    bool original_verbose       = verbosity_is_enabled();
72 #endif
73 
74    (void)ret;
75 
76 #ifndef IS_SALAMANDER
77 #if defined(HAVE_LOGGER)
78    logger_init();
79 #elif defined(HAVE_FILE_LOGGER)
80    retro_main_log_file_init("/retroarch-log.txt");
81 #endif
82 #endif
83 
84 #ifdef _XBOX360
85    /* Detect install environment. */
86    if (XContentGetLicenseMask(&license_mask, NULL) == ERROR_SUCCESS)
87    {
88       XContentQueryVolumeDeviceType("GAME",&volume_device_type, NULL);
89 
90       switch(volume_device_type)
91       {
92          case XCONTENTDEVICETYPE_HDD: /* Launched from content package on HDD */
93          case XCONTENTDEVICETYPE_MU:  /* Launched from content package on USB/Memory Unit. */
94          case XCONTENTDEVICETYPE_ODD: /* Launched from content package on Optial Disc Drive. */
95          default:                     /* Launched from content package on unknown device. */
96             break;
97       }
98    }
99 #endif
100 
101 #if defined(_XBOX1)
102    strcpy_literal(g_defaults.dirs[DEFAULT_DIR_CORE], "D:");
103    fill_pathname_join(g_defaults.path_config, g_defaults.dirs[DEFAULT_DIR_CORE],
104          FILE_PATH_MAIN_CONFIG, sizeof(g_defaults.path_config));
105    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_SAVESTATE],
106          g_defaults.dirs[DEFAULT_DIR_CORE],
107          "savestates",
108          sizeof(g_defaults.dirs[DEFAULT_DIR_SAVESTATE]));
109    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_SRAM],
110          g_defaults.dirs[DEFAULT_DIR_CORE],
111          "savefiles",
112          sizeof(g_defaults.dirs[DEFAULT_DIR_SRAM]));
113    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_SYSTEM],
114          g_defaults.dirs[DEFAULT_DIR_CORE],
115          "system",
116          sizeof(g_defaults.dirs[DEFAULT_DIR_SYSTEM]));
117    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_SCREENSHOT],
118          g_defaults.dirs[DEFAULT_DIR_CORE],
119          "screenshots",
120          sizeof(g_defaults.dirs[DEFAULT_DIR_SCREENSHOT]));
121    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_OVERLAY],
122          g_defaults.dirs[DEFAULT_DIR_CORE],
123          "overlays",
124          sizeof(g_defaults.dirs[DEFAULT_DIR_OVERLAY]));
125 #ifdef HAVE_VIDEO_LAYOUT
126    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_VIDEO_LAYOUT],
127          g_defaults.dirs[DEFAULT_DIR_CORE],
128          "layouts",
129          sizeof(g_defaults.dirs[DEFAULT_DIR_VIDEO_LAYOUT]));
130 #endif
131    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_ASSETS],
132          g_defaults.dirs[DEFAULT_DIR_CORE],
133          "media", sizeof(g_defaults.dirs[DEFAULT_DIR_ASSETS]));
134    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_CORE_ASSETS],
135          g_defaults.dirs[DEFAULT_DIR_CORE],
136          "downloads", sizeof(g_defaults.dirs[DEFAULT_DIR_CORE_ASSETS]));
137    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_PLAYLIST],
138          g_defaults.dirs[DEFAULT_DIR_CORE],
139          "playlists", sizeof(g_defaults.dirs[DEFAULT_DIR_PLAYLIST]));
140    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_LOGS],
141          g_defaults.dirs[DEFAULT_DIR_CORE],
142          "logs", sizeof(g_defaults.dirs[DEFAULT_DIR_LOGS]));
143 #elif defined(_XBOX360)
144    strcpy_literal(g_defaults.dirs[DEFAULT_DIR_CORE],
145          "game:");
146    strcpy_literal(g_defaults.path_config,
147          "game:\\retroarch.cfg");
148    strcpy_literal(g_defaults.dirs[DEFAULT_DIR_SCREENSHOT],
149          "game:");
150    strcpy_literal(g_defaults.dirs[DEFAULT_DIR_SAVESTATE],
151          "game:\\savestates");
152    strcpy_literal(g_defaults.dirs[DEFAULT_DIR_PLAYLIST],
153          "game:\\playlists");
154    strcpy_literal(g_defaults.dirs[DEFAULT_DIR_SRAM],
155          "game:\\savefiles");
156    strcpy_literal(g_defaults.dirs[DEFAULT_DIR_SYSTEM],
157          "game:\\system");
158    strcpy_literal(g_defaults.dirs[DEFAULT_DIR_LOGS],
159          "game:\\logs");
160 #endif
161    fill_pathname_join(g_defaults.dirs[DEFAULT_DIR_CORE_INFO],
162          g_defaults.dirs[DEFAULT_DIR_CORE],
163          "info", sizeof(g_defaults.dirs[DEFAULT_DIR_CORE_INFO]));
164 
165 #ifndef IS_SALAMANDER
166 #if defined(_XBOX1)
167    if (XGetLaunchInfo(&launch_type, &ptr) == ERROR_SUCCESS)
168    {
169       char *extracted_path = NULL;
170       if (launch_type == LDT_FROM_DEBUGGER_CMDLINE)
171          goto exit;
172 
173       extracted_path = (char*)&ptr.Data;
174 
175       if (
176             !string_is_empty(extracted_path)
177             && (!strstr(extracted_path, "Pool"))
178             /* Hack. Unknown problem */)
179       {
180          /* Auto-start game */
181          strlcpy(path, extracted_path, sizeof(path));
182       }
183    }
184 #elif defined(_XBOX360)
185    if (XGetLaunchDataSize(&dwLaunchDataSize) == ERROR_SUCCESS)
186    {
187       char *extracted_path                 = (char*)calloc(dwLaunchDataSize, sizeof(char));
188       BYTE* pLaunchData                    = (BYTE*)calloc(dwLaunchDataSize, sizeof(BYTE));
189 
190       XGetLaunchData(pLaunchData, dwLaunchDataSize);
191       memset(extracted_path, 0, dwLaunchDataSize);
192 
193       strlcpy(extracted_path, pLaunchData, dwLaunchDataSize);
194 
195       /* Auto-start game */
196       if (!string_is_empty(extracted_path))
197          strlcpy(path, extracted_path, sizeof(path));
198 
199       if (pLaunchData)
200          free(pLaunchData);
201    }
202 #endif
203    if (!string_is_empty(path))
204    {
205       struct rarch_main_wrap *args = (struct rarch_main_wrap*)params_data;
206 
207       if (args)
208       {
209          /* Auto-start game. */
210          args->touched        = true;
211          args->no_content     = false;
212          args->verbose        = false;
213          args->config_path    = NULL;
214          args->sram_path      = NULL;
215          args->state_path     = NULL;
216          args->content_path   = path;
217          args->libretro_path  = NULL;
218       }
219    }
220 #endif
221 
222 #ifndef IS_SALAMANDER
223 #ifdef _XBOX1
224 exit:
225    if (original_verbose)
226       verbosity_enable();
227    else
228       verbosity_disable();
229 #endif
230 
231    dir_check_defaults("custom.ini");
232 #endif
233 }
234 
frontend_xdk_init(void * data)235 static void frontend_xdk_init(void *data)
236 {
237    (void)data;
238 #if defined(_XBOX1) && !defined(IS_SALAMANDER)
239    /* Mount drives */
240    xbox_io_mount("A:", "cdrom0");
241    xbox_io_mount("C:", "Harddisk0\\Partition0");
242    xbox_io_mount("E:", "Harddisk0\\Partition1");
243    xbox_io_mount("Z:", "Harddisk0\\Partition2");
244    xbox_io_mount("F:", "Harddisk0\\Partition6");
245    xbox_io_mount("G:", "Harddisk0\\Partition7");
246 #endif
247 }
248 
frontend_xdk_exec(const char * path,bool should_load_content)249 static void frontend_xdk_exec(const char *path, bool should_load_content)
250 {
251 #ifndef IS_SALAMANDER
252    bool original_verbose       = verbosity_is_enabled();
253 #endif
254 #if defined(_XBOX1)
255 #ifndef IS_SALAMANDER
256    LAUNCH_DATA ptr;
257 #endif
258 #elif defined(_XBOX360)
259    char game_path[1024]        = {0};
260 #endif
261 
262 #ifdef IS_SALAMANDER
263    if (!string_is_empty(path))
264 #ifdef _XBOX360
265       XLaunchNewImage(path, 0);
266 #else
267       XLaunchNewImage(path, NULL);
268 #endif
269 #else
270 #if defined(_XBOX1)
271    memset(&ptr, 0, sizeof(ptr));
272 
273    if (should_load_content && !path_is_empty(RARCH_PATH_CONTENT))
274       snprintf((char*)ptr.Data, sizeof(ptr.Data), "%s", path_get(RARCH_PATH_CONTENT));
275 
276    if (!string_is_empty(path))
277       XLaunchNewImage(path, !string_is_empty((const char*)ptr.Data) ? &ptr : NULL);
278 #elif defined(_XBOX360)
279    if (should_load_content && !path_is_empty(RARCH_PATH_CONTENT))
280    {
281       strlcpy(game_path, path_get(RARCH_PATH_CONTENT), sizeof(game_path));
282       XSetLaunchData(game_path, MAX_LAUNCH_DATA_SIZE);
283    }
284 
285    if (!string_is_empty(path))
286       XLaunchNewImage(path, 0);
287 #endif
288 #endif
289 #ifndef IS_SALAMANDER
290    if (original_verbose)
291       verbosity_enable();
292    else
293       verbosity_disable();
294 #endif
295 }
296 
297 #ifndef IS_SALAMANDER
frontend_xdk_set_fork(enum frontend_fork fork_mode)298 static bool frontend_xdk_set_fork(enum frontend_fork fork_mode)
299 {
300    switch (fork_mode)
301    {
302       case FRONTEND_FORK_CORE:
303          RARCH_LOG("FRONTEND_FORK_CORE\n");
304          xdk_fork_mode  = fork_mode;
305          break;
306       case FRONTEND_FORK_CORE_WITH_ARGS:
307          RARCH_LOG("FRONTEND_FORK_CORE_WITH_ARGS\n");
308          xdk_fork_mode  = fork_mode;
309          break;
310       case FRONTEND_FORK_RESTART:
311          RARCH_LOG("FRONTEND_FORK_SALAMANDER_RESTART\n");
312          /* NOTE: We don't implement Salamander, so just turn
313           * this into FRONTEND_FORK_CORE. */
314          xdk_fork_mode  = FRONTEND_FORK_CORE;
315          break;
316       case FRONTEND_FORK_NONE:
317       default:
318          return false;
319    }
320 
321    return true;
322 }
323 #endif
324 
frontend_xdk_exitspawn(char * s,size_t len,char * args)325 static void frontend_xdk_exitspawn(char *s, size_t len, char *args)
326 {
327    bool should_load_content = false;
328 #ifndef IS_SALAMANDER
329    if (xdk_fork_mode == FRONTEND_FORK_NONE)
330       return;
331 
332    switch (xdk_fork_mode)
333    {
334       case FRONTEND_FORK_CORE_WITH_ARGS:
335          should_load_content = true;
336          break;
337       case FRONTEND_FORK_NONE:
338       default:
339          break;
340    }
341 #endif
342    frontend_xdk_exec(s, should_load_content);
343 }
344 
frontend_xdk_get_rating(void)345 static int frontend_xdk_get_rating(void)
346 {
347 #if defined(_XBOX360)
348    return 11;
349 #elif defined(_XBOX1)
350    return 7;
351 #endif
352 }
353 
frontend_xdk_get_arch(void)354 enum frontend_architecture frontend_xdk_get_arch(void)
355 {
356 #if defined(_XBOX360)
357    return FRONTEND_ARCH_PPC;
358 #elif defined(_XBOX1)
359    return FRONTEND_ARCH_X86;
360 #else
361    return FRONTEND_ARCH_NONE;
362 #endif
363 }
364 
frontend_xdk_parse_drive_list(void * data,bool load_content)365 static int frontend_xdk_parse_drive_list(void *data, bool load_content)
366 {
367 #ifndef IS_SALAMANDER
368    file_list_t *list = (file_list_t*)data;
369    enum msg_hash_enums enum_idx = load_content ?
370       MENU_ENUM_LABEL_FILE_DETECT_CORE_LIST_PUSH_DIR :
371       MENU_ENUM_LABEL_FILE_BROWSER_DIRECTORY;
372 
373 #if defined(_XBOX1)
374    menu_entries_append_enum(list,
375          "C:",
376          msg_hash_to_str(MENU_ENUM_LABEL_FILE_DETECT_CORE_LIST_PUSH_DIR),
377          enum_idx,
378          FILE_TYPE_DIRECTORY, 0, 0);
379    menu_entries_append_enum(list,
380          "D:",
381          msg_hash_to_str(MENU_ENUM_LABEL_FILE_DETECT_CORE_LIST_PUSH_DIR),
382          enum_idx,
383          FILE_TYPE_DIRECTORY, 0, 0);
384    menu_entries_append_enum(list,
385          "E:",
386          msg_hash_to_str(MENU_ENUM_LABEL_FILE_DETECT_CORE_LIST_PUSH_DIR),
387          enum_idx,
388          FILE_TYPE_DIRECTORY, 0, 0);
389    menu_entries_append_enum(list,
390          "F:",
391          msg_hash_to_str(MENU_ENUM_LABEL_FILE_DETECT_CORE_LIST_PUSH_DIR),
392          enum_idx,
393          FILE_TYPE_DIRECTORY, 0, 0);
394    menu_entries_append_enum(list,
395          "G:",
396          msg_hash_to_str(MENU_ENUM_LABEL_FILE_DETECT_CORE_LIST_PUSH_DIR),
397          enum_idx,
398          FILE_TYPE_DIRECTORY, 0, 0);
399 #elif defined(_XBOX360)
400    menu_entries_append_enum(list,
401          "game:",
402          msg_hash_to_str(MENU_ENUM_LABEL_FILE_DETECT_CORE_LIST_PUSH_DIR),
403          enum_idx,
404          FILE_TYPE_DIRECTORY, 0, 0);
405 #endif
406 #endif
407 
408    return 0;
409 }
410 
411 frontend_ctx_driver_t frontend_ctx_xdk = {
412    frontend_xdk_get_env_settings,/* env_settings */
413    frontend_xdk_init,            /* init   */
414    NULL,                         /* deinit */
415    frontend_xdk_exitspawn,       /* exitspawn */
416    NULL,                         /* process_args */
417    frontend_xdk_exec,            /* exec */
418 #ifdef IS_SALAMANDER
419    NULL,                         /* set_fork */
420 #else
421    frontend_xdk_set_fork,        /* set_fork */
422 #endif
423    NULL,                         /* shutdown */
424    NULL,                         /* get_name */
425    NULL,                         /* get_os */
426    frontend_xdk_get_rating,
427    NULL,                         /* content_loaded */
428    frontend_xdk_get_arch,        /* get_architecture */
429    NULL,                         /* get_powerstate */
430    frontend_xdk_parse_drive_list,/* parse_drive_list */
431    NULL,                         /* get_total_mem */
432    NULL,                         /* get_free_mem */
433    NULL,                         /* install_signal_handler */
434    NULL,                         /* get_sighandler_state */
435    NULL,                         /* set_sighandler_state */
436    NULL,                         /* destroy_sighandler_state */
437    NULL,                         /* attach_console */
438    NULL,                         /* detach_console */
439    NULL,                         /* get_lakka_version */
440    NULL,                         /* set_screen_brightness */
441    NULL,                         /* watch_path_for_changes */
442    NULL,                         /* check_for_path_changes */
443    NULL,                         /* set_sustained_performance_mode */
444    NULL,                         /* get_cpu_model_name */
445    NULL,                         /* get_user_language */
446    NULL,                         /* is_narrator_running */
447    NULL,                         /* accessibility_speak */
448    "xdk",                        /* ident */
449    NULL                          /* get_video_driver */
450 };
451