rockbox/apps/settings.c
Linus Nielsen Feltzing 478da628f0 Xing headers now support mono better, added the 'editable files' option
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@3572 a1c6a512-1295-4272-9138-f99709370657
2003-04-20 22:00:30 +00:00

2017 lines
67 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2002 by wavey@wavey.org
* RTC config saving code (C) 2002 by hessu@hes.iki.fi
*
* All files in this archive are subject to the GNU General Public License.
* See the file COPYING in the source tree root for full license agreement.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#include <stdio.h>
#include "config.h"
#include "kernel.h"
#include "thread.h"
#include "settings.h"
#include "disk.h"
#include "panic.h"
#include "debug.h"
#include "button.h"
#include "usb.h"
#include "backlight.h"
#include "lcd.h"
#include "mpeg.h"
#include "string.h"
#include "ata.h"
#include "fat.h"
#include "power.h"
#include "backlight.h"
#include "powermgmt.h"
#include "status.h"
#include "atoi.h"
#include "screens.h"
#include "ctype.h"
#include "file.h"
#include "errno.h"
#include "system.h"
#ifdef HAVE_LCD_BITMAP
#include "icons.h"
#include "font.h"
#include "peakmeter.h"
#endif
#include "lang.h"
#include "language.h"
#include "wps-display.h"
#include "powermgmt.h"
#include "sprintf.h"
#include "keyboard.h"
struct user_settings global_settings;
char rockboxdir[] = ROCKBOX_DIR; /* config/font/data file directory */
#define CONFIG_BLOCK_VERSION 4
#define CONFIG_BLOCK_SIZE 512
#define RTC_BLOCK_SIZE 44
#ifdef HAVE_LCD_BITMAP
#define MAX_LINES 10
#else
#define MAX_LINES 2
#endif
/********************************************
Config block as saved on the battery-packed RTC user RAM memory block
of 44 bytes, starting at offset 0x14 of the RTC memory space.
offset abs
0x00 0x14 "Roc" header signature: 0x52 0x6f 0x63
0x03 0x17 <version byte: 0x0>
0x04 0x18 <volume byte>
0x05 0x19 <balance byte>
0x06 0x1a <bass byte>
0x07 0x1b <treble byte>
0x08 0x1c <loudness byte>
0x09 0x1d <bass boost byte>
0x0a 0x1e <contrast (bit 0-5), invert bit (bit 6)>
0x0b 0x1f <backlight_on_when_charging, invert_cursor, backlight_timeout>
0x0c 0x20 <poweroff timer byte>
0x0d 0x21 <resume settings byte>
0x0e 0x22 <shuffle,dirfilter,sort_case,discharge,statusbar,show_hidden,
scroll bar>
0x0f 0x23 <volume type, battery type, timeformat, scroll speed>
0x10 0x24 <ff/rewind min step, acceleration rate>
0x11 0x25 <AVC, channel config>
0x12 0x26 <(int) Resume playlist index, or -1 if no playlist resume>
0x16 0x2a <(int) Byte offset into resume file>
0x1a 0x2e <time until disk spindown>
0x1b 0x2f <browse current, play selected, queue_resume>
0x1c 0x30 <peak meter hold timeout (bit 0-4),
rec_editable (bit 7)>
0x1d 0x31 <(int) queue resume index>
0x21 0x35 <repeat mode (bit 0-1), rec. channels (bit 2),
mic gain (bit 4-7)>
0x22 0x36 <rec. quality (bit 0-2), source (bit 3-4), frequency (bit 5-7)>
0x23 0x37 <rec. left gain (bit 0-3)>
0x24 0x38 <rec. right gain (bit 0-3)>
0x25 0x39 <disk poweroff flag (bit 0), MP3 buffer margin (bit 1-3),
Trickle charge flag (bit 4)>
0x26 0x40 <runtime low byte>
0x27 0x41 <runtime high byte>
0x28 0x42 <topruntime low byte>
0x29 0x43 <topruntime high byte>
0x2a <checksum 2 bytes: xor of 0x0-0x29>
Config memory is reset to 0xff and initialized with 'factory defaults' if
a valid header & checksum is not found. Config version number is only
increased when information is _relocated_ or space is _reused_ so that old
versions can read and modify configuration changed by new versions. New
versions should check for the value of '0xff' in each config memory
location used, and reset the setting in question with a factory default if
needed. Memory locations not used by a given version should not be
modified unless the header & checksum test fails.
Rest of config block, only saved to disk:
0xAE fade on pause/unpause/stop setting (bit 0)
0xB0 peak meter clip hold timeout (bit 0-4), peak meter performance (bit 7)
0xB1 peak meter release step size, peak_meter_dbfs (bit 7)
0xB2 peak meter min either in -db or in percent
0xB3 peak meter max either in -db or in percent
0xB4 battery capacity
0xB5 scroll step in pixels
0xB6 scroll start and endpoint delay
0xB7 bidir scroll setting (bidi if 0-200% longer than screen width)
0xB8 (char[20]) WPS file
0xCC (char[20]) Lang file
0xE0 (char[20]) Font file
0xF4 (int) Playlist first index
0xF8 (int) Playlist shuffle seed
0xFC (char[260]) Resume playlist (path/to/dir or path/to/playlist.m3u)
*************************************/
#include "rtc.h"
static unsigned char config_block[CONFIG_BLOCK_SIZE];
/*
* Calculates the checksum for the config block and returns it
*/
static unsigned short calculate_config_checksum(unsigned char* buf)
{
unsigned int i;
unsigned char cksum[2];
cksum[0] = cksum[1] = 0;
for (i=0; i < RTC_BLOCK_SIZE - 2; i+=2 ) {
cksum[0] ^= buf[i];
cksum[1] ^= buf[i+1];
}
return (cksum[0] << 8) | cksum[1];
}
/*
* initialize the config block buffer
*/
static void init_config_buffer( void )
{
DEBUGF( "init_config_buffer()\n" );
/* reset to 0xff - all unused */
memset(config_block, 0xff, CONFIG_BLOCK_SIZE);
/* insert header */
config_block[0] = 'R';
config_block[1] = 'o';
config_block[2] = 'c';
config_block[3] = CONFIG_BLOCK_VERSION;
}
/*
* save the config block buffer to disk or RTC RAM
*/
static int save_config_buffer( void )
{
unsigned short chksum;
#ifdef HAVE_RTC
unsigned int i;
#endif
DEBUGF( "save_config_buffer()\n" );
/* update the checksum in the end of the block before saving */
chksum = calculate_config_checksum(config_block);
config_block[ RTC_BLOCK_SIZE - 2 ] = chksum >> 8;
config_block[ RTC_BLOCK_SIZE - 1 ] = chksum & 0xff;
#ifdef HAVE_RTC
/* FIXME: okay, it _would_ be cleaner and faster to implement rtc_write so
that it would write a number of bytes at a time since the RTC chip
supports that, but this will have to do for now 8-) */
for (i=0; i < RTC_BLOCK_SIZE; i++ ) {
int r = rtc_write(0x14+i, config_block[i]);
if (r) {
DEBUGF( "save_config_buffer: rtc_write failed at addr 0x%02x: %d\n",
14+i, r );
return r;
}
}
#endif
if (fat_startsector() != 0)
ata_delayed_write( 61, config_block);
else
return -1;
return 0;
}
/*
* load the config block buffer from disk or RTC RAM
*/
static int load_config_buffer( void )
{
unsigned short chksum;
bool correct = false;
#ifdef HAVE_RTC
unsigned int i;
unsigned char rtc_block[RTC_BLOCK_SIZE];
#endif
DEBUGF( "load_config_buffer()\n" );
if (fat_startsector() != 0) {
ata_read_sectors( 61, 1, config_block);
/* calculate the checksum, check it and the header */
chksum = calculate_config_checksum(config_block);
if (config_block[0] == 'R' &&
config_block[1] == 'o' &&
config_block[2] == 'c' &&
config_block[3] == CONFIG_BLOCK_VERSION &&
(chksum >> 8) == config_block[RTC_BLOCK_SIZE - 2] &&
(chksum & 0xff) == config_block[RTC_BLOCK_SIZE - 1])
{
DEBUGF( "load_config_buffer: header & checksum test ok\n" );
correct = true;
}
}
#ifdef HAVE_RTC
/* read rtc block */
for (i=0; i < RTC_BLOCK_SIZE; i++ )
rtc_block[i] = rtc_read(0x14+i);
chksum = calculate_config_checksum(rtc_block);
/* if rtc block is ok, use that */
if (rtc_block[0] == 'R' &&
rtc_block[1] == 'o' &&
rtc_block[2] == 'c' &&
rtc_block[3] == CONFIG_BLOCK_VERSION &&
(chksum >> 8) == rtc_block[RTC_BLOCK_SIZE - 2] &&
(chksum & 0xff) == rtc_block[RTC_BLOCK_SIZE - 1])
{
memcpy(config_block, rtc_block, RTC_BLOCK_SIZE);
correct = true;
}
#endif
if ( !correct ) {
/* if checksum is not valid, clear the config buffer */
DEBUGF( "load_config_buffer: header & checksum test failed\n" );
init_config_buffer();
return -1;
}
return 0;
}
/*
* persist all runtime user settings to disk or RTC RAM
*/
int settings_save( void )
{
DEBUGF( "settings_save()\n" );
/* update the config block buffer with current
settings and save the block in the RTC */
config_block[0x4] = (unsigned char)global_settings.volume;
config_block[0x5] = (char)global_settings.balance;
config_block[0x6] = (unsigned char)global_settings.bass;
config_block[0x7] = (unsigned char)global_settings.treble;
config_block[0x8] = (unsigned char)global_settings.loudness;
config_block[0x9] = (unsigned char)global_settings.bass_boost;
config_block[0xa] = (unsigned char)
((global_settings.contrast & 0x3f) |
(global_settings.invert ? 0x40 : 0));
config_block[0xb] = (unsigned char)
((global_settings.backlight_on_when_charging?0x40:0) |
(global_settings.invert_cursor ? 0x20 : 0) |
(global_settings.backlight_timeout & 0x1f));
config_block[0xc] = (unsigned char)global_settings.poweroff;
config_block[0xd] = (unsigned char)global_settings.resume;
config_block[0xe] = (unsigned char)
((global_settings.playlist_shuffle & 1) |
((global_settings.dirfilter & 1) << 1) |
((global_settings.sort_case & 1) << 2) |
((global_settings.discharge & 1) << 3) |
((global_settings.statusbar & 1) << 4) |
((global_settings.dirfilter & 2) << 4) |
((global_settings.scrollbar & 1) << 6));
config_block[0xf] = (unsigned char)
((global_settings.volume_type & 1) |
((global_settings.battery_type & 1) << 1) |
((global_settings.timeformat & 1) << 2) |
( global_settings.scroll_speed << 3));
config_block[0x10] = (unsigned char)
((global_settings.ff_rewind_min_step & 15) << 4 |
(global_settings.ff_rewind_accel & 15));
config_block[0x11] = (unsigned char)
((global_settings.avc & 0x03) |
((global_settings.channel_config & 0x07) << 2));
memcpy(&config_block[0x12], &global_settings.resume_index, 4);
memcpy(&config_block[0x16], &global_settings.resume_offset, 4);
config_block[0x1a] = (unsigned char)global_settings.disk_spindown;
config_block[0x1b] = (unsigned char)
(((global_settings.browse_current & 1)) |
((global_settings.play_selected & 1) << 1) |
((global_settings.queue_resume & 3) << 2));
config_block[0x1c] = (unsigned char)global_settings.peak_meter_hold |
(global_settings.rec_editable?0x80:0);
memcpy(&config_block[0x1d], &global_settings.queue_resume_index, 4);
config_block[0x21] = (unsigned char)
((global_settings.repeat_mode & 3) |
((global_settings.rec_channels & 1) << 2) |
((global_settings.rec_mic_gain & 0x0f) << 4));
config_block[0x22] = (unsigned char)
((global_settings.rec_quality & 7) |
((global_settings.rec_source & 1) << 3) |
((global_settings.rec_frequency & 7) << 5));
config_block[0x23] = (unsigned char)global_settings.rec_left_gain;
config_block[0x24] = (unsigned char)global_settings.rec_right_gain;
config_block[0x25] = (unsigned char)
((global_settings.disk_poweroff & 1) |
((global_settings.buffer_margin & 7) << 1) |
((global_settings.trickle_charge & 1) << 4));
{
static long lasttime = 0;
global_settings.runtime += (current_tick - lasttime) / HZ;
lasttime = current_tick;
if ( global_settings.runtime > global_settings.topruntime )
global_settings.topruntime = global_settings.runtime;
config_block[0x26]=(unsigned char)(global_settings.runtime & 0xff);
config_block[0x27]=(unsigned char)(global_settings.runtime >> 8);
config_block[0x28]=(unsigned char)(global_settings.topruntime & 0xff);
config_block[0x29]=(unsigned char)(global_settings.topruntime >> 8);
}
config_block[0xae] = (unsigned char)global_settings.fade_on_stop;
config_block[0xb0] = (unsigned char)global_settings.peak_meter_clip_hold |
(global_settings.peak_meter_performance ? 0x80 : 0);
config_block[0xb1] = global_settings.peak_meter_release |
(global_settings.peak_meter_dbfs ? 0x80 : 0);
config_block[0xb2] = (unsigned char)global_settings.peak_meter_min;
config_block[0xb3] = (unsigned char)global_settings.peak_meter_max;
config_block[0xb4]=(global_settings.battery_capacity - 1000) / 50;
config_block[0xb5]=(unsigned char)global_settings.scroll_step;
config_block[0xb6]=(unsigned char)global_settings.scroll_delay;
config_block[0xb7]=(unsigned char)global_settings.bidir_limit;
strncpy(&config_block[0xb8], global_settings.wps_file, MAX_FILENAME);
strncpy(&config_block[0xcc], global_settings.lang_file, MAX_FILENAME);
strncpy(&config_block[0xe0], global_settings.font_file, MAX_FILENAME);
memcpy(&config_block[0xF4], &global_settings.resume_first_index, 4);
memcpy(&config_block[0xF8], &global_settings.resume_seed, 4);
strncpy(&config_block[0xFC], global_settings.resume_file, MAX_PATH);
DEBUGF( "+Resume file %s\n",global_settings.resume_file );
DEBUGF( "+Resume index %X offset %X\n",
global_settings.resume_index,
global_settings.resume_offset );
DEBUGF( "+Resume shuffle %s seed %X\n",
global_settings.playlist_shuffle?"on":"off",
global_settings.resume_seed );
if(save_config_buffer())
{
lcd_clear_display();
#ifdef HAVE_LCD_CHARCELLS
lcd_puts(0, 0, str(LANG_SETTINGS_SAVE_PLAYER));
lcd_puts(0, 1, str(LANG_SETTINGS_BATTERY_PLAYER));
#else
lcd_puts(4, 2, str(LANG_SETTINGS_SAVE_RECORDER));
lcd_puts(2, 4, str(LANG_SETTINGS_BATTERY_RECORDER));
lcd_update();
#endif
sleep(HZ*2);
return -1;
}
return 0;
}
#ifdef HAVE_LCD_BITMAP
/**
* Applies the range infos stored in global_settings to
* the peak meter.
*/
void settings_apply_pm_range(void)
{
int pm_min, pm_max;
/* depending on the scale mode (dBfs or percent) the values
of global_settings.peak_meter_dbfs have different meanings */
if (global_settings.peak_meter_dbfs)
{
/* convert to dBfs * 100 */
pm_min = -(((int)global_settings.peak_meter_min) * 100);
pm_max = -(((int)global_settings.peak_meter_max) * 100);
}
else
{
/* percent is stored directly -> no conversion */
pm_min = global_settings.peak_meter_min;
pm_max = global_settings.peak_meter_max;
}
/* apply the range */
peak_meter_init_range(global_settings.peak_meter_dbfs, pm_min, pm_max);
}
#endif /* HAVE_LCD_BITMAP */
void settings_apply(void)
{
char buf[64];
mpeg_sound_set(SOUND_BASS, global_settings.bass);
mpeg_sound_set(SOUND_TREBLE, global_settings.treble);
mpeg_sound_set(SOUND_BALANCE, global_settings.balance);
mpeg_sound_set(SOUND_VOLUME, global_settings.volume);
mpeg_sound_set(SOUND_CHANNELS, global_settings.channel_config);
#ifdef HAVE_MAS3587F
mpeg_sound_set(SOUND_LOUDNESS, global_settings.loudness);
mpeg_sound_set(SOUND_SUPERBASS, global_settings.bass_boost);
mpeg_sound_set(SOUND_AVC, global_settings.avc);
#endif
lcd_set_contrast(global_settings.contrast);
lcd_scroll_speed(global_settings.scroll_speed);
backlight_set_timeout(global_settings.backlight_timeout);
backlight_set_on_when_charging(global_settings.backlight_on_when_charging);
ata_spindown(global_settings.disk_spindown);
#ifdef HAVE_ATA_POWER_OFF
ata_poweroff(global_settings.disk_poweroff);
#endif
set_poweroff_timeout(global_settings.poweroff);
#ifdef HAVE_CHARGE_CTRL
charge_restart_level = global_settings.discharge ?
CHARGE_RESTART_LO : CHARGE_RESTART_HI;
enable_trickle_charge(global_settings.trickle_charge);
#endif
set_battery_capacity(global_settings.battery_capacity);
#ifdef HAVE_LCD_BITMAP
lcd_set_invert_display(global_settings.invert);
settings_apply_pm_range();
peak_meter_init_times(
global_settings.peak_meter_release, global_settings.peak_meter_hold,
global_settings.peak_meter_clip_hold);
#endif
if ( global_settings.wps_file[0] &&
global_settings.wps_file[0] != 0xff ) {
snprintf(buf, sizeof buf, ROCKBOX_DIR "/%s.wps",
global_settings.wps_file);
wps_load(buf, false);
}
else
wps_reset();
#ifdef HAVE_LCD_BITMAP
if ( global_settings.font_file[0] &&
global_settings.font_file[0] != 0xff ) {
snprintf(buf, sizeof buf, ROCKBOX_DIR "/%s.fnt",
global_settings.font_file);
font_load(buf);
}
else
font_reset();
lcd_scroll_step(global_settings.scroll_step);
#endif
lcd_bidir_scroll(global_settings.bidir_limit);
lcd_scroll_delay(global_settings.scroll_delay * (HZ/10));
if ( global_settings.lang_file[0] &&
global_settings.lang_file[0] != 0xff ) {
snprintf(buf, sizeof buf, ROCKBOX_DIR "/%s.lng",
global_settings.lang_file);
lang_load(buf);
}
}
/*
* load settings from disk or RTC RAM
*/
void settings_load(void)
{
DEBUGF( "reload_all_settings()\n" );
/* populate settings with default values */
settings_reset();
/* load the buffer from the RTC (resets it to all-unused if the block
is invalid) and decode the settings which are set in the block */
if (!load_config_buffer()) {
if (config_block[0x4] != 0xFF)
global_settings.volume = config_block[0x4];
if (config_block[0x5] != 0xFF)
global_settings.balance = (char)config_block[0x5];
if (config_block[0x6] != 0xFF)
global_settings.bass = config_block[0x6];
if (config_block[0x7] != 0xFF)
global_settings.treble = config_block[0x7];
if (config_block[0x8] != 0xFF)
global_settings.loudness = config_block[0x8];
if (config_block[0x9] != 0xFF)
global_settings.bass_boost = config_block[0x9];
if (config_block[0xa] != 0xFF) {
global_settings.contrast = config_block[0xa] & 0x3f;
global_settings.invert =
config_block[0xa] & 0x40 ? true : false;
if ( global_settings.contrast < MIN_CONTRAST_SETTING )
global_settings.contrast = DEFAULT_CONTRAST_SETTING;
}
if (config_block[0xb] != 0xFF) {
/* Bit 7 is unused to be able to detect uninitialized entry */
global_settings.backlight_timeout = config_block[0xb] & 0x1f;
global_settings.invert_cursor =
config_block[0xb] & 0x20 ? true : false;
global_settings.backlight_on_when_charging =
config_block[0xb] & 0x40 ? true : false;
}
if (config_block[0xc] != 0xFF)
global_settings.poweroff = config_block[0xc];
if (config_block[0xd] != 0xFF)
global_settings.resume = config_block[0xd];
if (config_block[0xe] != 0xFF) {
global_settings.playlist_shuffle = config_block[0xe] & 1;
global_settings.dirfilter = (config_block[0xe] >> 1) & 1;
global_settings.sort_case = (config_block[0xe] >> 2) & 1;
global_settings.discharge = (config_block[0xe] >> 3) & 1;
global_settings.statusbar = (config_block[0xe] >> 4) & 1;
global_settings.dirfilter |= ((config_block[0xe] >> 5) & 1) << 1;
global_settings.scrollbar = (config_block[0xe] >> 6) & 1;
/* Don't use the last bit, it must be unused to detect
an uninitialized entry */
}
if (config_block[0xf] != 0xFF) {
global_settings.volume_type = config_block[0xf] & 1;
global_settings.battery_type = (config_block[0xf] >> 1) & 1;
global_settings.timeformat = (config_block[0xf] >> 2) & 1;
global_settings.scroll_speed = config_block[0xf] >> 3;
}
if (config_block[0x10] != 0xFF) {
global_settings.ff_rewind_min_step = (config_block[0x10] >> 4) & 15;
global_settings.ff_rewind_accel = config_block[0x10] & 15;
}
if (config_block[0x11] != 0xFF)
{
global_settings.avc = config_block[0x11] & 0x03;
global_settings.channel_config = (config_block[0x11] >> 2) & 0x07;
}
if (config_block[0x12] != 0xFF)
memcpy(&global_settings.resume_index, &config_block[0x12], 4);
if (config_block[0x16] != 0xFF)
memcpy(&global_settings.resume_offset, &config_block[0x16], 4);
if (config_block[0x1a] != 0xFF)
global_settings.disk_spindown = config_block[0x1a];
if (config_block[0x1b] != 0xFF) {
global_settings.browse_current = (config_block[0x1b]) & 1;
global_settings.play_selected = (config_block[0x1b] >> 1) & 1;
global_settings.queue_resume = (config_block[0x1b] >> 2) & 3;
}
if (config_block[0x1c] != 0xFF) {
global_settings.peak_meter_hold = (config_block[0x1c]) & 0x1f;
global_settings.rec_editable =
config_block[0x1c]?true:false;
}
if (config_block[0x1d] != 0xFF)
memcpy(&global_settings.queue_resume_index, &config_block[0x1d],
4);
if (config_block[0x21] != 0xFF)
{
global_settings.repeat_mode = config_block[0x21] & 3;
global_settings.rec_channels = (config_block[0x21] >> 2) & 1;
global_settings.rec_mic_gain = (config_block[0x21] >> 4) & 0x0f;
}
if (config_block[0x22] != 0xFF)
{
global_settings.rec_quality = config_block[0x22] & 7;
global_settings.rec_source = (config_block[0x22] >> 3) & 3;
global_settings.rec_frequency = (config_block[0x22] >> 5) & 7;
}
if (config_block[0x23] != 0xFF)
global_settings.rec_left_gain = config_block[0x23] & 0x0f;
if (config_block[0x24] != 0xFF)
global_settings.rec_right_gain = config_block[0x24] & 0x0f;
if (config_block[0x25] != 0xFF)
{
global_settings.disk_poweroff = config_block[0x25] & 1;
global_settings.buffer_margin = (config_block[0x25] >> 1) & 7;
global_settings.trickle_charge = (config_block[0x25] >> 4) & 1;
}
if (config_block[0x27] != 0xff)
global_settings.runtime =
config_block[0x26] | (config_block[0x27] << 8);
if (config_block[0x29] != 0xff)
global_settings.topruntime =
config_block[0x28] | (config_block[0x29] << 8);
global_settings.fade_on_stop=config_block[0xae];
global_settings.peak_meter_clip_hold = (config_block[0xb0]) & 0x1f;
global_settings.peak_meter_performance =
(config_block[0xb0] & 0x80) != 0;
global_settings.peak_meter_release = config_block[0xb1] & 0x7f;
global_settings.peak_meter_dbfs = (config_block[0xb1] & 0x80) != 0;
global_settings.peak_meter_min = config_block[0xb2];
global_settings.peak_meter_max = config_block[0xb3];
global_settings.battery_capacity = config_block[0xb4]*50 + 1000;
if (config_block[0xb5] != 0xff)
global_settings.scroll_step = config_block[0xb5];
if (config_block[0xb6] != 0xff)
global_settings.scroll_delay = config_block[0xb6];
if (config_block[0xb7] != 0xff)
global_settings.bidir_limit = config_block[0xb7];
if (config_block[0xae] != 0xff)
global_settings.fade_on_stop = config_block[0xae];
memcpy(&global_settings.resume_first_index, &config_block[0xF4], 4);
memcpy(&global_settings.resume_seed, &config_block[0xF8], 4);
strncpy(global_settings.wps_file, &config_block[0xb8], MAX_FILENAME);
strncpy(global_settings.lang_file, &config_block[0xcc], MAX_FILENAME);
strncpy(global_settings.font_file, &config_block[0xe0], MAX_FILENAME);
strncpy(global_settings.resume_file, &config_block[0xFC], MAX_PATH);
global_settings.resume_file[MAX_PATH]=0;
}
settings_apply();
}
/* Read (up to) a line of text from fd into buffer and return number of bytes
* read (which may be larger than the number of bytes stored in buffer). If
* an error occurs, -1 is returned (and buffer contains whatever could be
* read). A line is terminated by a LF char. Neither LF nor CR chars are
* stored in buffer.
*/
static int read_line(int fd, char* buffer, int buffer_size)
{
int count = 0;
int num_read = 0;
errno = 0;
while (count < buffer_size)
{
unsigned char c;
if (1 != read(fd, &c, 1))
break;
num_read++;
if ( c == '\n' )
break;
if ( c == '\r' )
continue;
buffer[count++] = c;
}
buffer[MIN(count, buffer_size - 1)] = 0;
return errno ? -1 : num_read;
}
/* parse a line from a configuration file. the line format is:
setting name: setting value
Any whitespace before setting name or value (after ':') is ignored.
A # as first non-whitespace character discards the whole line.
Function sets pointers to null-terminated setting name and value.
Returns false if no valid config entry was found.
*/
static bool settings_parseline(char* line, char** name, char** value)
{
char* ptr;
while ( isspace(*line) )
line++;
if ( *line == '#' )
return false;
ptr = strchr(line, ':');
if ( !ptr )
return false;
*name = line;
*ptr = 0;
ptr++;
while (isspace(*ptr))
ptr++;
*value = ptr;
return true;
}
void set_file(char* filename, char* setting, int maxlen)
{
char* fptr = strrchr(filename,'/');
int len;
int extlen = 0;
char* ptr;
if (!fptr)
return;
*fptr = 0;
fptr++;
len = strlen(fptr);
ptr = fptr + len;
while (*ptr != '.') {
extlen++;
ptr--;
}
if (strcmp(ROCKBOX_DIR, filename) || (len-extlen > maxlen))
return;
strncpy(setting, fptr, len-extlen);
setting[len-extlen]=0;
settings_save();
}
static void set_sound(char* value, int type, int* setting)
{
int num = atoi(value);
num = mpeg_phys2val(type, num);
if ((num > mpeg_sound_max(type)) ||
(num < mpeg_sound_min(type)))
{
num = mpeg_sound_default(type);
}
*setting = num;
mpeg_sound_set(type, num);
#ifdef HAVE_MAS3507D
/* This is required to actually apply balance */
if (SOUND_BALANCE == type)
mpeg_sound_set(SOUND_VOLUME, global_settings.volume);
#endif
}
static void set_cfg_bool(bool* variable, char* value)
{
/* look for the 'n' in 'on' */
if ((value[1] & 0xdf) == 'N')
*variable = true;
else
*variable = false;
}
static void set_cfg_int(int* variable, char* value, int min, int max )
{
*variable = atoi(value);
if (*variable < min)
*variable = min;
else
if (*variable > max)
*variable = max;
}
static void set_cfg_option(int* variable, char* value,
char* options[], int numoptions )
{
int i;
for (i=0; i<numoptions; i++) {
if (!strcasecmp(options[i], value)) {
*variable = i;
break;
}
}
}
bool settings_load_config(char* file)
{
int fd;
char line[128];
fd = open(file, O_RDONLY);
if (fd < 0)
return false;
while (read_line(fd, line, sizeof line) > 0)
{
char* name;
char* value;
if (!settings_parseline(line, &name, &value))
continue;
if (!strcasecmp(name, "volume"))
set_sound(value, SOUND_VOLUME, &global_settings.volume);
else if (!strcasecmp(name, "bass"))
set_sound(value, SOUND_BASS, &global_settings.bass);
else if (!strcasecmp(name, "treble"))
set_sound(value, SOUND_TREBLE, &global_settings.treble);
else if (!strcasecmp(name, "balance"))
set_sound(value, SOUND_BALANCE, &global_settings.balance);
else if (!strcasecmp(name, "channels")) {
static char* options[] = {
"stereo","stereo narrow","mono","mono left",
"mono right","karaoke","stereo wide"};
set_cfg_option(&global_settings.channel_config, value,
options, 7);
}
else if (!strcasecmp(name, "wps")) {
if (wps_load(value,false))
set_file(value, global_settings.wps_file, MAX_FILENAME);
}
else if (!strcasecmp(name, "lang")) {
if (!lang_load(value))
set_file(value, global_settings.lang_file, MAX_FILENAME);
}
else if (!strcasecmp(name, "bidir limit"))
set_cfg_int(&global_settings.bidir_limit, value, 0, 200);
#ifdef HAVE_LCD_BITMAP
else if (!strcasecmp(name, "font")) {
if (font_load(value))
set_file(value, global_settings.font_file, MAX_FILENAME);
}
else if (!strcasecmp(name, "scroll step"))
set_cfg_int(&global_settings.scroll_step, value, 1, LCD_WIDTH);
else if (!strcasecmp(name, "statusbar"))
set_cfg_bool(&global_settings.statusbar, value);
else if (!strcasecmp(name, "peak meter release"))
set_cfg_int(&global_settings.peak_meter_release, value, 1, 0x7e);
else if (!strcasecmp(name, "peak meter hold")) {
static char* options[] = {
"off","200ms","300ms","500ms",
"1","2","3","4","5","6","7","8","9","10",
"15","20","30","1min"};
set_cfg_option(&global_settings.peak_meter_hold, value,
options, 18);
}
else if (!strcasecmp(name, "peak meter clip hold")) {
static char* options[] = {
"on","1","2","3","4","5","6","7","8","9","10",
"15","20","25","30","45","60","90",
"2min","3min","5min","10min","20min","45min","90min"};
set_cfg_option(&global_settings.peak_meter_clip_hold, value,
options, 25);
}
else if (!strcasecmp(name, "peak meter dbfs"))
set_cfg_bool(&global_settings.peak_meter_dbfs, value);
else if (!strcasecmp(name, "peak meter min"))
set_cfg_int(&global_settings.peak_meter_min, value, 0, 100);
else if (!strcasecmp(name, "peak meter max"))
set_cfg_int(&global_settings.peak_meter_max, value, 0, 100);
else if (!strcasecmp(name, "peak meter busy"))
set_cfg_bool(&global_settings.peak_meter_performance, value);
else if (!strcasecmp(name, "volume display")) {
static char* options[] = {"graphic", "numeric"};
set_cfg_option(&global_settings.volume_type, value, options, 2);
}
else if (!strcasecmp(name, "battery display")) {
static char* options[] = {"graphic", "numeric"};
set_cfg_option(&global_settings.battery_type, value, options, 2);
}
else if (!strcasecmp(name, "time format")) {
static char* options[] = {"24hour", "12hour"};
set_cfg_option(&global_settings.timeformat, value, options, 2);
}
else if (!strcasecmp(name, "scrollbar"))
set_cfg_bool(&global_settings.scrollbar, value);
else if (!strcasecmp(name, "invert"))
set_cfg_bool(&global_settings.invert, value);
else if (!strcasecmp(name, "invert cursor"))
set_cfg_bool(&global_settings.invert_cursor, value);
#endif
else if (!strcasecmp(name, "shuffle"))
set_cfg_bool(&global_settings.playlist_shuffle, value);
else if (!strcasecmp(name, "repeat")) {
static char* options[] = {"off", "all", "one"};
set_cfg_option(&global_settings.repeat_mode, value, options, 3);
}
else if (!strcasecmp(name, "resume")) {
static char* options[] = {"off", "ask", "ask once", "on"};
set_cfg_option(&global_settings.resume, value, options, 4);
}
else if (!strcasecmp(name, "sort case"))
set_cfg_bool(&global_settings.sort_case, value);
else if (!strcasecmp(name, "show files")) {
static char* options[] = {"all", "supported","music", "playlists"};
set_cfg_option(&global_settings.dirfilter, value, options, 4);
}
else if (!strcasecmp(name, "follow playlist"))
set_cfg_bool(&global_settings.browse_current, value);
else if (!strcasecmp(name, "play selected"))
set_cfg_bool(&global_settings.play_selected, value);
else if (!strcasecmp(name, "contrast"))
set_cfg_int(&global_settings.contrast, value,
0, MAX_CONTRAST_SETTING);
else if (!strcasecmp(name, "scroll speed"))
set_cfg_int(&global_settings.scroll_speed, value, 1, 10);
else if (!strcasecmp(name, "scan min step")) {
static char* options[] =
{"1","2","3","4","5","6","8","10",
"15","20","25","30","45","60"};
set_cfg_option(&global_settings.ff_rewind_min_step, value,
options, 14);
}
else if (!strcasecmp(name, "scan accel"))
set_cfg_int(&global_settings.ff_rewind_accel, value, 0, 15);
else if (!strcasecmp(name, "scroll delay"))
set_cfg_int(&global_settings.scroll_delay, value, 0, 250);
else if (!strcasecmp(name, "backlight timeout")) {
static char* options[] = {
"off","on","1","2","3","4","5","6","7","8","9",
"10","15","20","25","30","45","60","90"};
set_cfg_option(&global_settings.backlight_timeout, value,
options, 19);
}
else if (!strcasecmp(name, "backlight when plugged"))
set_cfg_bool(&global_settings.backlight_on_when_charging, value);
else if (!strcasecmp(name, "antiskip"))
set_cfg_int(&global_settings.buffer_margin, value, 0, 7);
else if (!strcasecmp(name, "disk spindown"))
set_cfg_int(&global_settings.disk_spindown, value, 3, 254);
#ifdef HAVE_ATA_POWER_OFF
else if (!strcasecmp(name, "disk poweroff"))
set_cfg_bool(&global_settings.disk_poweroff, value);
#endif
#ifdef HAVE_MAS3587F
else if (!strcasecmp(name, "loudness"))
set_sound(value, SOUND_LOUDNESS, &global_settings.loudness);
else if (!strcasecmp(name, "bass boost"))
set_sound(value, SOUND_SUPERBASS, &global_settings.bass_boost);
else if (!strcasecmp(name, "auto volume")) {
static char* options[] = {"off", "2", "4", "8" };
set_cfg_option(&global_settings.avc, value, options, 4);
}
else if (!strcasecmp(name, "rec mic gain"))
set_sound(value, SOUND_MIC_GAIN, &global_settings.rec_mic_gain);
else if (!strcasecmp(name, "rec left gain"))
set_sound(value, SOUND_LEFT_GAIN, &global_settings.rec_left_gain);
else if (!strcasecmp(name, "rec right gain"))
set_sound(value, SOUND_RIGHT_GAIN, &global_settings.rec_right_gain);
else if (!strcasecmp(name, "rec quality"))
set_cfg_int(&global_settings.rec_quality, value, 0, 7);
else if (!strcasecmp(name, "rec source")) {
static char* options[] = {"mic", "line", "spdif"};
set_cfg_option(&global_settings.rec_source, value, options, 3);
}
else if (!strcasecmp(name, "rec frequency")) {
static char* options[] = {"44", "48", "32", "22", "24", "16"};
set_cfg_option(&global_settings.rec_frequency, value, options, 6);
}
else if (!strcasecmp(name, "rec channels")) {
static char* options[] = {"stereo", "mono"};
set_cfg_option(&global_settings.rec_channels, value, options, 2);
}
else if (!strcasecmp(name, "editable recordings")) {
set_cfg_bool(&global_settings.rec_editable, value);
}
#endif
else if (!strcasecmp(name, "idle poweroff")) {
static char* options[] = {"off","1","2","3","4","5","6","7","8",
"9","10","15","30","45","60"};
set_cfg_option(&global_settings.poweroff, value, options, 15);
}
else if (!strcasecmp(name, "battery capacity"))
set_cfg_int(&global_settings.battery_capacity, value,
1500, BATTERY_CAPACITY_MAX);
#ifdef HAVE_CHARGE_CTRL
else if (!strcasecmp(name, "deep discharge"))
set_cfg_bool(&global_settings.discharge, value);
else if (!strcasecmp(name, "trickle charge"))
set_cfg_bool(&global_settings.trickle_charge, value);
#endif
else if (!strcasecmp(name, "volume fade"))
set_cfg_bool(&global_settings.fade_on_stop, value);
}
close(fd);
settings_apply();
settings_save();
return true;
}
bool settings_save_config(void)
{
bool done = false;
int fd, i, value;
char buf[MAX_PATH + 32];
char filename[MAX_PATH];
/* find unused filename */
for (i=0; ; i++) {
snprintf(filename, sizeof filename, "/.rockbox/config%02d.cfg", i);
fd = open(filename, O_RDONLY);
if (fd < 0)
break;
close(fd);
}
/* allow user to modify filename */
while (!done) {
if (!kbd_input(filename, sizeof filename)) {
fd = creat(filename,0);
if (fd < 0) {
lcd_clear_display();
lcd_puts(0,0,str(LANG_FAILED));
lcd_update();
sleep(HZ);
}
else
done = true;
}
else
break;
}
/* abort if file couldn't be created */
if (!done) {
lcd_clear_display();
lcd_puts(0,0,str(LANG_RESET_DONE_CANCEL));
lcd_update();
sleep(HZ);
return false;
}
strncpy(buf, "# >>> .cfg file created by rockbox <<<\r\n", sizeof(buf));
write(fd, buf, strlen(buf));
strncpy(buf, "# >>> http://rockbox.haxx.se <<<\r\n#\r\n", sizeof(buf));
write(fd, buf, strlen(buf));
snprintf(buf, sizeof(buf), "#\r\n# wps / language / font \r\n#\r\n");
write(fd, buf, strlen(buf));
if (global_settings.wps_file[0] != 0) {
snprintf(buf, sizeof(buf),
"wps: /.rockbox/%s.wps\r\n", global_settings.wps_file);
write(fd, buf, strlen(buf));
}
if (global_settings.lang_file[0] != 0) {
snprintf(buf, sizeof(buf), "lang: /.rockbox/%s.lng\r\n",
global_settings.lang_file);
write(fd, buf, strlen(buf));
}
#ifdef HAVE_LCD_BITMAP
if (global_settings.font_file[0] != 0) {
snprintf(buf, sizeof(buf), "font: /.rockbox/%s.fnt\r\n",
global_settings.font_file);
write(fd, buf, strlen(buf));
}
#endif
snprintf(buf, sizeof(buf), "#\r\n# Sound settings\r\n#\r\n");
write(fd, buf, strlen(buf));
value = mpeg_val2phys(SOUND_VOLUME, global_settings.volume);
snprintf(buf, sizeof(buf), "volume: %d\r\n", value);
write(fd, buf, strlen(buf));
value = mpeg_val2phys(SOUND_BASS, global_settings.bass);
snprintf(buf, sizeof(buf), "bass: %d\r\n", value);
write(fd, buf, strlen(buf));
value = mpeg_val2phys(SOUND_TREBLE, global_settings.treble);
snprintf(buf, sizeof(buf), "treble: %d\r\n", value);
write(fd, buf, strlen(buf));
value = mpeg_val2phys(SOUND_BALANCE, global_settings.balance);
snprintf(buf, sizeof(buf), "balance: %d\r\n", value);
write(fd, buf, strlen(buf));
{
static char* options[] =
{"stereo","stereo narrow","mono","mono left",
"mono right","karaoke","stereo wide"};
snprintf(buf, sizeof(buf), "channels: %s\r\n",
options[global_settings.channel_config]);
write(fd, buf, strlen(buf));
}
#ifdef HAVE_MAS3587F
value = mpeg_val2phys(SOUND_LOUDNESS, global_settings.loudness);
snprintf(buf, sizeof(buf), "loudness: %d\r\n", value);
write(fd, buf, strlen(buf));
value = mpeg_val2phys(SOUND_SUPERBASS, global_settings.bass_boost);
snprintf(buf, sizeof(buf), "bass boost: %d\r\n", value);
write(fd, buf, strlen(buf));
{
static char* options[] = {"off", "2", "4", "8" };
snprintf(buf, sizeof(buf), "auto volume: %s\r\n",
options[global_settings.avc]);
write(fd, buf, strlen(buf));
}
#endif
snprintf(buf, sizeof(buf), "#\r\n# Playback\r\n#\r\n");
write(fd, buf, strlen(buf));
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf), "shuffle: %s\r\n",
options[global_settings.playlist_shuffle]);
write(fd, buf, strlen(buf));
}
{
static char* options[] = {"off", "all", "one"};
snprintf(buf, sizeof(buf), "repeat: %s\r\n",
options[global_settings.repeat_mode]);
write(fd, buf, strlen(buf));
}
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf), "play selected: %s\r\n",
options[global_settings.play_selected]);
write(fd, buf, strlen(buf));
}
{
static char* options[] = {"off", "ask", "ask once", "on"};
snprintf(buf, sizeof(buf), "resume: %s\r\n",
options[global_settings.resume]);
write(fd, buf, strlen(buf));
}
{
static char* options[] =
{"1","2","3","4","5","6","8","10",
"15","20","25","30","45","60"};
snprintf(buf, sizeof(buf), "scan min step: %s\r\n",
options[global_settings.ff_rewind_min_step]);
write(fd, buf, strlen(buf));
}
snprintf(buf, sizeof(buf), "scan accel: %d\r\nantiskip: %d\r\n",
global_settings.ff_rewind_accel,
global_settings.buffer_margin);
write(fd, buf, strlen(buf));
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf), "volume fade: %s\r\n",
options[global_settings.fade_on_stop]);
write(fd, buf, strlen(buf));
}
snprintf(buf, sizeof(buf), "#\r\n# File View\r\n#\r\n");
write(fd, buf, strlen(buf));
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf), "sort case: %s\r\n",
options[global_settings.sort_case]);
write(fd, buf, strlen(buf));
}
{
static char* options[] =
{"all", "supported","music", "playlists"};
snprintf(buf, sizeof(buf), "show files: %s\r\n",
options[global_settings.dirfilter]);
write(fd, buf, strlen(buf));
}
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf), "follow playlist: %s\r\n",
options[global_settings.browse_current]);
write(fd, buf, strlen(buf));
}
snprintf(buf, sizeof(buf), "#\r\n# Display\r\n#\r\n");
write(fd, buf, strlen(buf));
#ifdef HAVE_LCD_BITMAP
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf), "statusbar: %s\r\nscrollbar: %s\r\n",
options[global_settings.statusbar],
options[global_settings.scrollbar]);
write(fd, buf, strlen(buf));
}
{
static char* options[] = {"graphic", "numeric"};
snprintf(buf, sizeof(buf),
"volume display: %s\r\nbattery display: %s\r\n",
options[global_settings.volume_type],
options[global_settings.battery_type]);
write(fd, buf, strlen(buf));
}
#endif
snprintf(buf, sizeof(buf), "scroll speed: %d\r\nscroll delay: %d\r\n",
global_settings.scroll_speed,
global_settings.scroll_delay);
write(fd, buf, strlen(buf));
#ifdef HAVE_LCD_BITMAP
snprintf(buf, sizeof(buf), "scroll step: %d\r\n",
global_settings.scroll_step);
write(fd, buf, strlen(buf));
#endif
snprintf(buf, sizeof(buf), "bidir limit: %d\r\n",
global_settings.bidir_limit);
write(fd, buf, strlen(buf));
{
static char* options[] =
{"off","on","1","2","3","4","5","6","7","8","9",
"10","15","20","25","30","45","60","90"};
snprintf(buf, sizeof(buf), "backlight timeout: %s\r\n",
options[global_settings.backlight_timeout]);
write(fd, buf, strlen(buf));
}
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf), "backlight when plugged: %s\r\n",
options[global_settings.backlight_on_when_charging]);
write(fd, buf, strlen(buf));
}
snprintf(buf, sizeof(buf), "contrast: %d\r\n", global_settings.contrast);
write(fd, buf, strlen(buf));
#ifdef HAVE_LCD_BITMAP
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf), "invert: %s\r\n",
options[global_settings.invert]);
write(fd, buf, strlen(buf));
snprintf(buf, sizeof(buf), "invert cursor: %s\r\n",
options[global_settings.invert_cursor]);
write(fd, buf, strlen(buf));
}
snprintf(buf, sizeof(buf), "peak meter release: %d\r\n",
global_settings.peak_meter_release);
write(fd, buf, strlen(buf));
{
static char* options[] =
{"off","200ms","300ms","500ms","1","2","3","4","5",
"6","7","8","9","10","15","20","30","1min"};
snprintf(buf, sizeof(buf), "peak meter hold: %s\r\n",
options[global_settings.peak_meter_hold]);
write(fd, buf, strlen(buf));
}
{
static char* options[] =
{"on","1","2","3","4","5","6","7","8","9","10","15","20","25","30",
"45","60","90","2min","3min","5min","10min","20min","45min","90min"};
snprintf(buf, sizeof(buf), "peak meter clip hold: %s\r\n",
options[global_settings.peak_meter_clip_hold]);
write(fd, buf, strlen(buf));
}
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf),
"peak meter busy: %s\r\npeak meter dbfs: %s\r\n",
options[global_settings.peak_meter_performance],
options[global_settings.peak_meter_dbfs]);
write(fd, buf, strlen(buf));
}
snprintf(buf, sizeof(buf), "peak meter min: %d\r\npeak meter max: %d\r\n",
global_settings.peak_meter_min,
global_settings.peak_meter_max);
write(fd, buf, strlen(buf));
#endif
snprintf(buf, sizeof(buf), "#\r\n# System\r\n#\r\ndisk spindown: %d\r\n",
global_settings.disk_spindown);
write(fd, buf, strlen(buf));
#ifdef HAVE_ATA_POWER_OFF
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf), "disk poweroff: %s\r\n",
options[global_settings.disk_poweroff]);
write(fd, buf, strlen(buf));
}
#endif
snprintf(buf, sizeof(buf), "battery capacity: %d\r\n",
global_settings.battery_capacity);
write(fd, buf, strlen(buf));
#ifdef HAVE_CHARGE_CTRL
{
static char* options[] = {"off","on"};
snprintf(buf, sizeof(buf),
"deep discharge: %s\r\ntrickle charge: %s\r\n",
options[global_settings.discharge],
options[global_settings.trickle_charge]);
write(fd, buf, strlen(buf));
}
#endif
#ifdef HAVE_LCD_BITMAP
{
static char* options[] = {"24hour", "12hour"};
snprintf(buf, sizeof(buf), "time format: %s\r\n",
options[global_settings.timeformat]);
write(fd, buf, strlen(buf));
}
#endif
{
static char* options[] =
{"off","1","2","3","4","5","6","7","8",
"9","10","15","30","45","60"};
snprintf(buf, sizeof(buf), "idle poweroff: %s\r\n",
options[global_settings.poweroff]);
write(fd, buf, strlen(buf));
}
#ifdef HAVE_MAS3587F
snprintf(buf, sizeof(buf), "#\r\n# Recording\r\n#\r\n");
write(fd, buf, strlen(buf));
snprintf(buf, sizeof(buf), "rec quality: %d\r\n",
global_settings.rec_quality);
write(fd, buf, strlen(buf));
{
static char* options[] = {"44", "48", "32", "22", "24", "16"};
snprintf(buf, sizeof(buf), "rec frequency: %s\r\n",
options[global_settings.rec_frequency]);
write(fd, buf, strlen(buf));
}
{
static char* options[] = {"mic", "line", "spdif"};
snprintf(buf, sizeof(buf), "rec source: %s\r\n",
options[global_settings.rec_source]);
write(fd, buf, strlen(buf));
}
{
static char* options[] = {"stereo", "mono"};
snprintf(buf, sizeof(buf), "rec channels: %s\r\n",
options[global_settings.rec_channels]);
write(fd, buf, strlen(buf));
}
snprintf(buf, sizeof(buf),
"rec mic gain: %d\r\nrec left gain: %d\r\nrec right gain: %d\r\n",
global_settings.rec_mic_gain,
global_settings.rec_left_gain,
global_settings.rec_right_gain);
write(fd, buf, strlen(buf));
{
static char* options[] = {"off", "on"};
snprintf(buf, sizeof(buf), "editable recordings: %s\r\n",
options[global_settings.rec_editable]);
write(fd, buf, strlen(buf));
}
#endif
close(fd);
lcd_clear_display();
lcd_puts(0,0,str(LANG_SETTINGS_SAVED1));
lcd_puts(0,1,str(LANG_SETTINGS_SAVED2));
lcd_update();
sleep(HZ);
return true;
}
/*
* reset all settings to their default value
*/
void settings_reset(void) {
DEBUGF( "settings_reset()\n" );
global_settings.volume = mpeg_sound_default(SOUND_VOLUME);
global_settings.balance = mpeg_sound_default(SOUND_BALANCE);
global_settings.bass = mpeg_sound_default(SOUND_BASS);
global_settings.treble = mpeg_sound_default(SOUND_TREBLE);
global_settings.loudness = mpeg_sound_default(SOUND_LOUDNESS);
global_settings.bass_boost = mpeg_sound_default(SOUND_SUPERBASS);
global_settings.avc = mpeg_sound_default(SOUND_AVC);
global_settings.channel_config = mpeg_sound_default(SOUND_CHANNELS);
global_settings.rec_quality = 5;
global_settings.rec_source = 0; /* 0=mic */
global_settings.rec_frequency = 0; /* 0=44.1kHz */
global_settings.rec_channels = 0; /* 0=Stereo */
global_settings.rec_mic_gain = 8;
global_settings.rec_left_gain = 2; /* 0dB */
global_settings.rec_right_gain = 2; /* 0dB */
global_settings.rec_editable = false;
global_settings.resume = RESUME_ASK;
global_settings.contrast = DEFAULT_CONTRAST_SETTING;
global_settings.invert = DEFAULT_INVERT_SETTING;
global_settings.poweroff = DEFAULT_POWEROFF_SETTING;
global_settings.backlight_timeout = DEFAULT_BACKLIGHT_TIMEOUT_SETTING;
global_settings.invert_cursor = DEFAULT_INVERT_CURSOR_SETTING;
global_settings.backlight_on_when_charging =
DEFAULT_BACKLIGHT_ON_WHEN_CHARGING_SETTING;
global_settings.battery_capacity = 1500; /* mAh */
global_settings.trickle_charge = true;
global_settings.dirfilter = SHOW_MUSIC;
global_settings.sort_case = false;
global_settings.statusbar = true;
global_settings.scrollbar = true;
global_settings.repeat_mode = REPEAT_ALL;
global_settings.playlist_shuffle = false;
global_settings.discharge = 0;
global_settings.timeformat = 0;
global_settings.volume_type = 0;
global_settings.battery_type = 0;
global_settings.scroll_speed = 8;
global_settings.bidir_limit = 50;
global_settings.scroll_delay = 100;
global_settings.scroll_step = 6;
global_settings.ff_rewind_min_step = DEFAULT_FF_REWIND_MIN_STEP;
global_settings.ff_rewind_accel = DEFAULT_FF_REWIND_ACCEL_SETTING;
global_settings.resume_index = -1;
global_settings.resume_offset = -1;
global_settings.save_queue_resume = true;
global_settings.queue_resume = 0;
global_settings.queue_resume_index = -1;
global_settings.disk_spindown = 5;
global_settings.disk_poweroff = false;
global_settings.buffer_margin = 0;
global_settings.browse_current = false;
global_settings.play_selected = true;
global_settings.peak_meter_release = 8;
global_settings.peak_meter_hold = 3;
global_settings.peak_meter_clip_hold = 16;
global_settings.peak_meter_dbfs = true;
global_settings.peak_meter_min = 60;
global_settings.peak_meter_max = 0;
global_settings.peak_meter_performance = false;
global_settings.wps_file[0] = 0;
global_settings.font_file[0] = 0;
global_settings.lang_file[0] = 0;
global_settings.runtime = 0;
global_settings.topruntime = 0;
global_settings.fade_on_stop = true;
}
/*
* dump the list of current settings
*/
void settings_display(void)
{
#ifdef DEBUG
DEBUGF( "\nsettings_display()\n" );
DEBUGF( "\nvolume:\t\t%d\nbalance:\t%d\nbass:\t\t%d\ntreble:\t\t%d\n"
"loudness:\t%d\nbass boost:\t%d\n",
global_settings.volume,
global_settings.balance,
global_settings.bass,
global_settings.treble,
global_settings.loudness,
global_settings.bass_boost );
DEBUGF( "contrast:\t%d\ninvert:\t%d\npoweroff:\t%d\nbacklight_timeout:\t%d\n",
global_settings.contrast,
global_settings.invert,
global_settings.poweroff,
global_settings.backlight_timeout );
#endif
}
bool set_bool(char* string, bool* variable )
{
return set_bool_options(string, variable, str(LANG_SET_BOOL_YES),
str(LANG_SET_BOOL_NO));
}
bool set_bool_options(char* string, bool* variable,
char* yes_str, char* no_str )
{
char* names[] = { no_str, yes_str };
int value = *variable;
bool result;
result = set_option(string, &value, names, 2, NULL);
*variable = value;
return result;
}
bool set_int(char* string,
char* unit,
int* variable,
void (*function)(int),
int step,
int min,
int max )
{
bool done = false;
int button;
int org_value=*variable;
#ifdef HAVE_LCD_BITMAP
if(global_settings.statusbar)
lcd_setmargins(0, STATUSBAR_HEIGHT);
else
lcd_setmargins(0, 0);
#endif
lcd_clear_display();
lcd_puts_scroll(0, 0, string);
while (!done) {
char str[32];
snprintf(str,sizeof str,"%d %s ", *variable, unit);
lcd_puts(0, 1, str);
#ifdef HAVE_LCD_BITMAP
status_draw();
#endif
lcd_update();
button = button_get_w_tmo(HZ/2);
switch(button) {
#ifdef HAVE_RECORDER_KEYPAD
case BUTTON_UP:
case BUTTON_UP | BUTTON_REPEAT:
#else
case BUTTON_RIGHT:
case BUTTON_RIGHT | BUTTON_REPEAT:
#endif
*variable += step;
break;
#ifdef HAVE_RECORDER_KEYPAD
case BUTTON_DOWN:
case BUTTON_DOWN | BUTTON_REPEAT:
#else
case BUTTON_LEFT:
case BUTTON_LEFT | BUTTON_REPEAT:
#endif
*variable -= step;
break;
#ifdef HAVE_RECORDER_KEYPAD
case BUTTON_LEFT:
case BUTTON_PLAY:
#else
case BUTTON_PLAY:
#endif
done = true;
break;
#ifdef HAVE_RECORDER_KEYPAD
case BUTTON_OFF:
#else
case BUTTON_STOP:
case BUTTON_MENU:
#endif
if (*variable != org_value) {
*variable=org_value;
lcd_stop_scroll();
lcd_puts(0, 0, str(LANG_MENU_SETTING_CANCEL));
lcd_update();
sleep(HZ/2);
}
done = true;
break;
case SYS_USB_CONNECTED:
usb_screen();
return true;
}
if(*variable > max )
*variable = max;
if(*variable < min )
*variable = min;
if ( function && button != BUTTON_NONE)
function(*variable);
}
lcd_stop_scroll();
return false;
}
bool set_option(char* string, int* variable, char* options[],
int numoptions, void (*function)(int))
{
bool done = false;
int button;
int org_value=*variable;
#ifdef HAVE_LCD_BITMAP
if(global_settings.statusbar)
lcd_setmargins(0, STATUSBAR_HEIGHT);
else
lcd_setmargins(0, 0);
#endif
lcd_clear_display();
lcd_puts_scroll(0, 0, string);
while ( !done ) {
lcd_puts(0, 1, options[*variable]);
#ifdef HAVE_LCD_BITMAP
status_draw();
#endif
lcd_update();
button = button_get_w_tmo(HZ/2);
switch (button) {
#ifdef HAVE_RECORDER_KEYPAD
case BUTTON_UP:
case BUTTON_UP | BUTTON_REPEAT:
#else
case BUTTON_RIGHT:
case BUTTON_RIGHT | BUTTON_REPEAT:
#endif
if ( *variable < (numoptions-1) )
(*variable)++;
else
(*variable) -= (numoptions-1);
break;
#ifdef HAVE_RECORDER_KEYPAD
case BUTTON_DOWN:
case BUTTON_DOWN | BUTTON_REPEAT:
#else
case BUTTON_LEFT:
case BUTTON_LEFT | BUTTON_REPEAT:
#endif
if ( *variable > 0 )
(*variable)--;
else
(*variable) += (numoptions-1);
break;
#ifdef HAVE_RECORDER_KEYPAD
case BUTTON_LEFT:
case BUTTON_PLAY:
#else
case BUTTON_PLAY:
#endif
done = true;
break;
#ifdef HAVE_RECORDER_KEYPAD
case BUTTON_OFF:
#else
case BUTTON_STOP:
case BUTTON_MENU:
#endif
if (*variable != org_value) {
*variable=org_value;
lcd_stop_scroll();
lcd_puts(0, 0, str(LANG_MENU_SETTING_CANCEL));
lcd_update();
sleep(HZ/2);
}
done = true;
break;
case SYS_USB_CONNECTED:
usb_screen();
return true;
}
if ( function && button != BUTTON_NONE)
function(*variable);
}
lcd_stop_scroll();
return false;
}
#ifdef HAVE_LCD_BITMAP
#define INDEX_X 0
#define INDEX_Y 1
#define INDEX_WIDTH 2
bool set_time(char* string, int timedate[])
{
bool done = false;
int button;
int min = 0, steps = 0;
int cursorpos = 0;
int lastcursorpos = !cursorpos;
unsigned char buffer[19];
int realyear;
int julianday;
int i;
unsigned char reffub[5];
unsigned int width, height;
unsigned int separator_width, weekday_width;
unsigned int line_height, prev_line_height;
char *dayname[] = {str(LANG_WEEKDAY_SUNDAY),
str(LANG_WEEKDAY_MONDAY),
str(LANG_WEEKDAY_TUESDAY),
str(LANG_WEEKDAY_WEDNESDAY),
str(LANG_WEEKDAY_THURSDAY),
str(LANG_WEEKDAY_FRIDAY),
str(LANG_WEEKDAY_SATURDAY)};
char *monthname[] = {str(LANG_MONTH_JANUARY),
str(LANG_MONTH_FEBRUARY),
str(LANG_MONTH_MARCH),
str(LANG_MONTH_APRIL),
str(LANG_MONTH_MAY),
str(LANG_MONTH_JUNE),
str(LANG_MONTH_JULY),
str(LANG_MONTH_AUGUST),
str(LANG_MONTH_SEPTEMBER),
str(LANG_MONTH_OCTOBER),
str(LANG_MONTH_NOVEMBER),
str(LANG_MONTH_DECEMBER)};
char cursor[][3] = {{ 0, 8, 12}, {18, 8, 12}, {36, 8, 12},
{24, 16, 24}, {54, 16, 18}, {78, 16, 12}};
char daysinmonth[] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
int monthname_len = 0, dayname_len = 0;
#ifdef HAVE_LCD_BITMAP
if(global_settings.statusbar)
lcd_setmargins(0, STATUSBAR_HEIGHT);
else
lcd_setmargins(0, 0);
#endif
lcd_clear_display();
lcd_puts_scroll(0, 0, string);
while ( !done ) {
/* calculate the number of days in febuary */
realyear = timedate[3] + 2000;
if((realyear % 4 == 0 && !(realyear % 100 == 0)) || realyear % 400 == 0)
daysinmonth[1] = 29;
else
daysinmonth[1] = 28;
/* fix day if month or year changed */
if (timedate[5] > daysinmonth[timedate[4] - 1])
timedate[5] = daysinmonth[timedate[4] - 1];
/* calculate day of week */
julianday = 0;
for(i = 0; i < timedate[4] - 1; i++) {
julianday += daysinmonth[i];
}
julianday += timedate[5];
timedate[6] = (realyear + julianday + (realyear - 1) / 4 -
(realyear - 1) / 100 + (realyear - 1) / 400 + 7 - 1) % 7;
snprintf(buffer, sizeof(buffer), "%02d:%02d:%02d ",
timedate[0], timedate[1], timedate[2]);
lcd_puts(0, 1, buffer);
/* recalculate the positions and offsets */
lcd_getstringsize(string, &width, &prev_line_height);
lcd_getstringsize(buffer, &width, &line_height);
lcd_getstringsize(":", &separator_width, &height);
/* hour */
strncpy(reffub, buffer, 2);
reffub[2] = '\0';
lcd_getstringsize(reffub, &width, &height);
cursor[0][INDEX_X] = 0;
cursor[0][INDEX_Y] = prev_line_height;
cursor[0][INDEX_WIDTH] = width;
/* minute */
strncpy(reffub, buffer + 3, 2);
reffub[2] = '\0';
lcd_getstringsize(reffub, &width, &height);
cursor[1][INDEX_X] = cursor[0][INDEX_WIDTH] + separator_width;
cursor[1][INDEX_Y] = prev_line_height;
cursor[1][INDEX_WIDTH] = width;
/* second */
strncpy(reffub, buffer + 6, 2);
reffub[2] = '\0';
lcd_getstringsize(reffub, &width, &height);
cursor[2][INDEX_X] = cursor[0][INDEX_WIDTH] + separator_width +
cursor[1][INDEX_WIDTH] + separator_width;
cursor[2][INDEX_Y] = prev_line_height;
cursor[2][INDEX_WIDTH] = width;
lcd_getstringsize(buffer, &width, &prev_line_height);
snprintf(buffer, sizeof(buffer), "%s 20%02d %s %02d ",
dayname[timedate[6]], timedate[3], monthname[timedate[4] - 1],
timedate[5]);
lcd_puts(0, 2, buffer);
/* recalculate the positions and offsets */
lcd_getstringsize(buffer, &width, &line_height);
/* store these 2 to prevent _repeated_ strlen calls */
monthname_len = strlen(monthname[timedate[4] - 1]);
dayname_len = strlen(dayname[timedate[6]]);
/* weekday */
strncpy(reffub, buffer, dayname_len);
reffub[dayname_len] = '\0';
lcd_getstringsize(reffub, &weekday_width, &height);
lcd_getstringsize(" ", &separator_width, &height);
/* year */
strncpy(reffub, buffer + dayname_len + 1, 4);
reffub[4] = '\0';
lcd_getstringsize(reffub, &width, &height);
cursor[3][INDEX_X] = weekday_width + separator_width;
cursor[3][INDEX_Y] = cursor[0][INDEX_Y] + prev_line_height;
cursor[3][INDEX_WIDTH] = width;
/* month */
strncpy(reffub, buffer + dayname_len + 6, monthname_len);
reffub[monthname_len] = '\0';
lcd_getstringsize(reffub, &width, &height);
cursor[4][INDEX_X] = weekday_width + separator_width +
cursor[3][INDEX_WIDTH] + separator_width;
cursor[4][INDEX_Y] = cursor[0][INDEX_Y] + prev_line_height;
cursor[4][INDEX_WIDTH] = width;
/* day */
strncpy(reffub, buffer + dayname_len + monthname_len + 7, 2);
reffub[2] = '\0';
lcd_getstringsize(reffub, &width, &height);
cursor[5][INDEX_X] = weekday_width + separator_width +
cursor[3][INDEX_WIDTH] + separator_width +
cursor[4][INDEX_WIDTH] + separator_width;
cursor[5][INDEX_Y] = cursor[0][INDEX_Y] + prev_line_height;
cursor[5][INDEX_WIDTH] = width;
lcd_invertrect(cursor[cursorpos][INDEX_X],
cursor[cursorpos][INDEX_Y] + lcd_getymargin(),
cursor[cursorpos][INDEX_WIDTH],
line_height);
lcd_puts(0, 4, str(LANG_TIME_SET));
lcd_puts(0, 5, str(LANG_TIME_REVERT));
#ifdef HAVE_LCD_BITMAP
status_draw();
#endif
lcd_update();
/* calculate the minimum and maximum for the number under cursor */
if(cursorpos!=lastcursorpos) {
lastcursorpos=cursorpos;
switch(cursorpos) {
case 0: /* hour */
min = 0;
steps = 24;
break;
case 1: /* minute */
case 2: /* second */
min = 0;
steps = 60;
break;
case 3: /* year */
min = 0;
steps = 100;
break;
case 4: /* month */
min = 1;
steps = 12;
break;
case 5: /* day */
min = 1;
steps = daysinmonth[timedate[4] - 1];
break;
}
}
button = button_get_w_tmo(HZ/2);
switch ( button ) {
case BUTTON_LEFT:
cursorpos = (cursorpos + 6 - 1) % 6;
break;
case BUTTON_RIGHT:
cursorpos = (cursorpos + 6 + 1) % 6;
break;
case BUTTON_UP:
case BUTTON_UP | BUTTON_REPEAT:
timedate[cursorpos] = (timedate[cursorpos] + steps - min + 1) %
steps + min;
if(timedate[cursorpos] == 0)
timedate[cursorpos] += min;
break;
case BUTTON_DOWN:
case BUTTON_DOWN | BUTTON_REPEAT:
timedate[cursorpos]=(timedate[cursorpos]+steps - min - 1) %
steps + min;
if(timedate[cursorpos] == 0)
timedate[cursorpos] += min;
break;
case BUTTON_ON:
done = true;
if (timedate[6] == 0) /* rtc needs 1 .. 7 */
timedate[6] = 7;
break;
case BUTTON_OFF:
done = true;
timedate[0] = -1;
break;
#ifdef HAVE_RECORDER_KEYPAD
case BUTTON_F3:
#ifdef HAVE_LCD_BITMAP
global_settings.statusbar = !global_settings.statusbar;
settings_save();
if(global_settings.statusbar)
lcd_setmargins(0, STATUSBAR_HEIGHT);
else
lcd_setmargins(0, 0);
lcd_clear_display();
lcd_puts_scroll(0, 0, string);
#endif
break;
#endif
case SYS_USB_CONNECTED:
usb_screen();
return true;
}
}
return false;
}
#endif