1 /***************************************************************************
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
10 * Copyright (C) 2002 Björn Stenberg
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation; either version 2
15 * of the License, or (at your option) any later version.
17 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
18 * KIND, either express or implied.
20 ****************************************************************************/
23 #include "gcc_extensions.h"
34 #include "filetypes.h"
39 #include "powermgmt.h"
46 #include "mp3_playback.h"
49 #include "backlight.h"
51 #include "debug_menu.h"
56 #include "core_alloc.h"
59 #include "usb_screen.h"
72 #include "eeprom_settings.h"
73 #include "scrobbler.h"
76 #include "skin_engine/skin_engine.h"
77 #include "statusbar-skinned.h"
78 #include "bootchart.h"
79 #if (CONFIG_PLATFORM & PLATFORM_ANDROID)
80 #include "notification.h"
82 #include "shortcuts.h"
84 #ifdef IPOD_ACCESSORY_PROTOCOL
88 #if (CONFIG_CODEC == SWCODEC)
92 #if (CONFIG_CODEC == SWCODEC) && defined(HAVE_RECORDING) && !defined(SIMULATOR)
93 #include "pcm_record.h"
97 #define SETTINGS_RESET BUTTON_REC
98 #elif (CONFIG_KEYPAD == GIGABEAT_PAD)
99 #define SETTINGS_RESET BUTTON_A
105 #if (CONFIG_STORAGE & STORAGE_MMC)
109 #ifdef HAVE_REMOTE_LCD
110 #include "lcd-remote.h"
113 #if CONFIG_USBOTG == USBOTG_ISP1362
117 #if CONFIG_USBOTG == USBOTG_M5636
121 #ifdef HAVE_HARDWARE_CLICK
125 #if (CONFIG_PLATFORM & PLATFORM_NATIVE)
126 #define MAIN_NORETURN_ATTR NORETURN_ATTR
128 /* gcc adds an implicit 'return 0;' at the end of main(), causing a warning
129 * with noreturn attribute */
130 #define MAIN_NORETURN_ATTR
133 #if (CONFIG_PLATFORM & (PLATFORM_SDL|PLATFORM_MAEMO|PLATFORM_PANDORA))
135 #include "sim_tasks.h"
137 #include "system-sdl.h"
138 #define HAVE_ARGV_MAIN
139 /* Don't use SDL_main on windows -> no more stdio redirection */
145 /*#define AUTOROCK*/ /* define this to check for "autostart.rock" on boot */
147 static void init(void);
148 /* main(), and various functions called by main() and init() may be
149 * be INIT_ATTR. These functions must not be called after the final call
150 * to root_menu() at the end of main()
151 * see definition of INIT_ATTR in config.h */
152 #ifdef HAVE_ARGV_MAIN
153 int main(int argc
, char *argv
[]) INIT_ATTR MAIN_NORETURN_ATTR
;
154 int main(int argc
, char *argv
[])
156 sys_handle_argv(argc
, argv
);
158 int main(void) INIT_ATTR MAIN_NORETURN_ATTR
;
167 screens
[i
].clear_display();
170 #ifdef HAVE_LCD_BITMAP
174 /* Keep the order of this 3
175 * Must be done before any code uses the multi-screen API */
177 /* All threads should be created and public queues registered by now */
178 usb_start_monitoring();
183 char filename
[MAX_PATH
];
191 if(file_exists(file
)) /* no complaint if it doesn't exist */
193 plugin_load(file
, NULL
); /* start if it does */
196 #endif /* #ifdef AUTOROCK */
198 global_status
.last_volume_change
= 0;
199 /* no calls INIT_ATTR functions after this point anymore!
200 * see definition of INIT_ATTR in config.h */
205 static int init_dircache(bool preinit
) INIT_ATTR
;
206 static int init_dircache(bool preinit
)
215 if (!global_settings
.dircache
)
218 # ifdef HAVE_EEPROM_SETTINGS
219 if (firmware_settings
.initialized
&& firmware_settings
.disk_clean
222 result
= dircache_load();
226 firmware_settings
.disk_clean
= false;
227 if (global_status
.dircache_size
<= 0)
229 /* This will be in default language, settings are not
230 applied yet. Not really any easy way to fix that. */
231 splash(0, str(LANG_SCANNING_DISK
));
235 dircache_build(global_status
.dircache_size
);
244 if (!dircache_is_enabled()
245 && !dircache_is_initializing())
247 if (global_status
.dircache_size
<= 0)
249 splash(0, str(LANG_SCANNING_DISK
));
252 result
= dircache_build(global_status
.dircache_size
);
257 /* Initialization of dircache failed. Manual action is
258 * necessary to enable dircache again.
260 splashf(0, "Dircache failed, disabled. Result: %d", result
);
261 global_settings
.dircache
= false;
269 global_status
.dircache_size
= dircache_get_cache_size();
281 static void init_tagcache(void) INIT_ATTR
;
282 static void init_tagcache(void)
285 #if CONFIG_CODEC == SWCODEC
286 long talked_tick
= 0;
290 while (!tagcache_is_initialized())
292 int ret
= tagcache_get_commit_step();
296 #if CONFIG_CODEC == SWCODEC
297 /* hwcodec can't use voice here, as the database commit
298 * uses the audio buffer. */
299 if(global_settings
.talk_menu
301 || TIME_AFTER(current_tick
, talked_tick
+7*HZ
)))
303 talked_tick
= current_tick
;
304 talk_id(LANG_TAGCACHE_INIT
, false);
305 talk_number(ret
, true);
306 talk_id(VOICE_OF
, true);
307 talk_number(tagcache_get_max_commit_step(), true);
310 #ifdef HAVE_LCD_BITMAP
313 splashf(0, "[%d/%d] %s", ret
, tagcache_get_max_commit_step(),
314 str(LANG_TAGCACHE_INIT
));
318 splashf(0, "%s [%d/%d]", str(LANG_TAGCACHE_INIT
), ret
,
319 tagcache_get_max_commit_step());
322 lcd_double_height(false);
323 lcd_putsf(0, 1, " DB [%d/%d]", ret
,
324 tagcache_get_max_commit_step());
341 #if (CONFIG_PLATFORM & PLATFORM_HOSTED)
343 static void init(void)
346 core_allocator_init();
353 #ifdef HAVE_REMOTE_LCD
356 #ifdef HAVE_LCD_BITMAP
358 global_status
.font_id
[i
] = FONT_SYSFIXED
;
368 #if (CONFIG_PLATFORM & PLATFORM_ANDROID)
371 lang_init(core_language_builtin
, language_strings
,
372 LANG_LAST_INDEX_IN_ARRAY
);
379 /* Keep the order of this 3 (viewportmanager handles statusbars)
380 * Must be done before any code uses the multi-screen API */
381 gui_syncstatusbar_init(&statusbars
);
382 gui_sync_skin_init();
384 viewportmanager_init();
388 settings_load(SETTINGS_ALL
);
389 settings_apply(true);
391 init_dircache(false);
400 #if CONFIG_CODEC != SWCODEC
401 mp3_init( global_settings
.volume
,
402 global_settings
.bass
,
403 global_settings
.treble
,
404 global_settings
.balance
,
405 global_settings
.loudness
,
407 global_settings
.channel_config
,
408 global_settings
.stereo_width
,
409 global_settings
.mdb_strength
,
410 global_settings
.mdb_harmonics
,
411 global_settings
.mdb_center
,
412 global_settings
.mdb_shape
,
413 global_settings
.mdb_enable
,
414 global_settings
.superbass
);
415 #endif /* CONFIG_CODEC != SWCODEC */
421 #if (CONFIG_CODEC == SWCODEC) && defined(HAVE_RECORDING)
425 settings_apply_skins();
430 static void init(void) INIT_ATTR
;
431 static void init(void)
434 bool mounted
= false;
435 #if CONFIG_CHARGING && (CONFIG_CPU == SH7034)
436 /* if nobody initialized ATA before, I consider this a cold start */
437 bool coldstart
= (PACR2
& 0x4000) != 0; /* starting from Flash */
441 core_allocator_init();
444 #ifdef HAVE_ADJUSTABLE_CPU_FREQ
445 set_cpu_frequency(CPUFREQ_NORMAL
);
447 coldfire_set_pllcr_audio_bits(DEFAULT_PLLCR_AUDIO_BITS
);
460 /* current_tick should be ticking by now */
464 #ifdef HAVE_REMOTE_LCD
467 #ifdef HAVE_LCD_BITMAP
469 global_status
.font_id
[i
] = FONT_SYSFIXED
;
478 lang_init(core_language_builtin
, language_strings
,
479 LANG_LAST_INDEX_IN_ARRAY
);
493 CHART(">settings_load(RTC)");
494 settings_load(SETTINGS_RTC
); /* early load parts of global_settings */
495 CHART("<settings_load(RTC)");
501 #if CONFIG_USBOTG == USBOTG_ISP1362
503 #elif CONFIG_USBOTG == USBOTG_M5636
511 /* Don't initialize power management here if it could incorrectly
512 * measure battery voltage, and it's not needed for charging. */
513 #if !defined(NEED_ATA_POWER_BATT_MEASURE) || \
514 (CONFIG_CHARGING > CHARGING_MONITOR)
522 #ifdef HAVE_HARDWARE_CLICK
526 /* Keep the order of this 3 (viewportmanager handles statusbars)
527 * Must be done before any code uses the multi-screen API */
528 CHART(">gui_syncstatusbar_init");
529 gui_syncstatusbar_init(&statusbars
);
530 CHART("<gui_syncstatusbar_init");
531 CHART(">sb_skin_init");
533 CHART("<sb_skin_init");
534 CHART(">gui_sync_wps_init");
535 gui_sync_skin_init();
536 CHART("<gui_sync_wps_init");
537 CHART(">viewportmanager_init");
538 viewportmanager_init();
539 CHART("<viewportmanager_init");
541 #if CONFIG_CHARGING && (CONFIG_CPU == SH7034)
542 /* charger_inserted() can't be used here because power_thread()
543 hasn't checked power_input_status() yet */
544 if (coldstart
&& (power_input_status() & POWER_INPUT_MAIN_CHARGER
)
545 && !global_settings
.car_adapter_mode
546 #ifdef ATA_POWER_PLAYERSTYLE
547 && !ide_powered() /* relies on probing result from bootloader */
551 rc
= charging_screen(); /* display a "charging" screen */
552 if (rc
== 1) /* charger removed */
554 /* "On" pressed or USB connected: proceed */
555 show_logo(); /* again, to provide better visual feedback */
560 disk_init_subsystem();
561 CHART(">storage_init");
563 CHART("<storage_init");
566 #ifdef HAVE_LCD_BITMAP
568 lcd_putsf(0, 1, "ATA error: %d", rc
);
569 lcd_puts(0, 3, "Press ON to debug");
571 while(!(button_get(true) & BUTTON_REL
)); /* DO NOT CHANGE TO ACTION SYSTEM */
574 panicf("ata: %d", rc
);
577 #if defined(NEED_ATA_POWER_BATT_MEASURE) && \
578 (CONFIG_CHARGING <= CHARGING_MONITOR)
579 /* After storage_init(), ATA power must be on, so battery voltage
580 * can be measured. Initialize power management if it was delayed. */
583 #ifdef HAVE_EEPROM_SETTINGS
584 CHART(">eeprom_settings_init");
585 eeprom_settings_init();
586 CHART("<eeprom_settings_init");
589 #ifndef HAVE_USBSTACK
590 usb_start_monitoring();
591 while (usb_detect() == USB_INSERTED
)
593 #ifdef HAVE_EEPROM_SETTINGS
594 firmware_settings
.disk_clean
= false;
596 /* enter USB mode early, before trying to mount */
597 if (button_get_w_tmo(HZ
/10) == SYS_USB_CONNECTED
)
598 #if (CONFIG_STORAGE & STORAGE_MMC)
599 if (!mmc_touched() ||
600 (mmc_remove_request() == SYS_HOTSWAP_EXTRACTED
))
603 gui_usb_screen_run(true);
604 mounted
= true; /* mounting done @ end of USB mode */
606 #ifdef HAVE_USB_POWER
607 if (usb_powered()) /* avoid deadlock */
615 CHART(">disk_mount_all");
616 rc
= disk_mount_all();
617 CHART("<disk_mount_all");
621 lcd_puts(0, 0, "No partition");
622 lcd_puts(0, 1, "found.");
623 #ifdef HAVE_LCD_BITMAP
624 lcd_puts(0, 2, "Insert USB cable");
625 lcd_puts(0, 3, "and fix it.");
629 while(button_get(true) != SYS_USB_CONNECTED
) {};
630 gui_usb_screen_run(true);
635 #if defined(SETTINGS_RESET) || (CONFIG_KEYPAD == IPOD_4G_PAD) || \
636 (CONFIG_KEYPAD == IRIVER_H10_PAD)
637 #ifdef SETTINGS_RESET
638 /* Reset settings if holding the reset button. (Rec on Archos,
640 if ((button_status() & SETTINGS_RESET
) == SETTINGS_RESET
)
642 /* Reset settings if the hold button is turned on */
646 splash(HZ
*2, str(LANG_RESET_DONE_CLEAR
));
652 CHART(">settings_load(ALL)");
653 settings_load(SETTINGS_ALL
);
654 CHART("<settings_load(ALL)");
657 CHART(">init_dircache(true)");
658 rc
= init_dircache(true);
659 CHART("<init_dircache(true)");
663 remove(TAGCACHE_STATEFILE
);
667 CHART(">settings_apply(true)");
668 settings_apply(true);
669 CHART("<settings_apply(true)");
670 CHART(">init_dircache(false)");
671 init_dircache(false);
672 CHART("<init_dircache(false)");
674 CHART(">init_tagcache");
676 CHART("<init_tagcache");
679 #ifdef HAVE_EEPROM_SETTINGS
680 if (firmware_settings
.initialized
)
682 /* In case we crash. */
683 firmware_settings
.disk_clean
= false;
684 CHART(">eeprom_settings_store");
685 eeprom_settings_store();
686 CHART("<eeprom_settings_store");
695 #if CONFIG_CODEC != SWCODEC
696 /* No buffer allocation (see buffer.c) may take place after the call to
697 audio_init() since the mpeg thread takes the rest of the buffer space */
698 mp3_init( global_settings
.volume
,
699 global_settings
.bass
,
700 global_settings
.treble
,
701 global_settings
.balance
,
702 global_settings
.loudness
,
704 global_settings
.channel_config
,
705 global_settings
.stereo_width
,
706 global_settings
.mdb_strength
,
707 global_settings
.mdb_harmonics
,
708 global_settings
.mdb_center
,
709 global_settings
.mdb_shape
,
710 global_settings
.mdb_enable
,
711 global_settings
.superbass
);
712 #endif /* CONFIG_CODEC != SWCODEC */
714 CHART(">audio_init");
716 CHART("<audio_init");
718 #if (CONFIG_CODEC == SWCODEC) && defined(HAVE_RECORDING)
722 /* runtime database has to be initialized after audio_init() */
726 car_adapter_mode_init();
728 #ifdef IPOD_ACCESSORY_PROTOCOL
729 iap_setup(global_settings
.serial_bitrate
);
731 #ifdef HAVE_ACCESSORY_SUPPLY
732 accessory_supply_set(global_settings
.accessory_supply
);
734 #ifdef HAVE_LINEOUT_POWEROFF
735 lineout_set(global_settings
.lineout_active
);
737 #ifdef HAVE_HOTSWAP_STORAGE_AS_MAIN
738 CHART("<check_bootfile(false)");
739 check_bootfile(false); /* remember write time and filesize */
740 CHART(">check_bootfile(false)");
742 CHART("<settings_apply_skins");
743 settings_apply_skins();
744 CHART(">settings_apply_skins");
748 void cop_main(void) MAIN_NORETURN_ATTR
;
751 /* This is the entry point for the coprocessor
752 Anyone not running an upgraded bootloader will never reach this point,
753 so it should not be assumed that the coprocessor be usable even on
754 platforms which support it.
756 A kernel thread is initially setup on the coprocessor and immediately
757 destroyed for purposes of continuity. The cop sits idle until at least
758 one thread exists on it. */
763 /* This should never be reached */
771 #endif /* SIMULATOR */