Search
lxdream.org :: lxdream/src/config.c
lxdream 0.9.1
released Jun 29
Download Now
filename src/config.c
changeset 866:86cd01c2b2d3
prev759:f16975739abc
next1015:ad448bedc48a
author nkeynes
date Tue Oct 14 08:44:37 2008 +0000 (15 years ago)
permissions -rw-r--r--
last change Fix a few more subtle flag problems
file annotate diff log raw
nkeynes@450
     1
/**
nkeynes@561
     2
 * $Id$
nkeynes@450
     3
 *
nkeynes@450
     4
 * User configuration support
nkeynes@450
     5
 *
nkeynes@450
     6
 * Copyright (c) 2005 Nathan Keynes.
nkeynes@450
     7
 *
nkeynes@450
     8
 * This program is free software; you can redistribute it and/or modify
nkeynes@450
     9
 * it under the terms of the GNU General Public License as published by
nkeynes@450
    10
 * the Free Software Foundation; either version 2 of the License, or
nkeynes@450
    11
 * (at your option) any later version.
nkeynes@450
    12
 *
nkeynes@450
    13
 * This program is distributed in the hope that it will be useful,
nkeynes@450
    14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
nkeynes@450
    15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
nkeynes@450
    16
 * GNU General Public License for more details.
nkeynes@450
    17
 */
nkeynes@450
    18
nkeynes@450
    19
#include <unistd.h>
nkeynes@450
    20
#include <stdio.h>
nkeynes@450
    21
#include <errno.h>
nkeynes@450
    22
#include <stdlib.h>
nkeynes@450
    23
#include <string.h>
nkeynes@480
    24
#include <glib/gmem.h>
nkeynes@450
    25
#include <glib/gstrfuncs.h>
nkeynes@450
    26
#include "dream.h"
nkeynes@450
    27
#include "config.h"
nkeynes@450
    28
#include "maple/maple.h"
nkeynes@450
    29
bhaal22@643
    30
bhaal22@643
    31
extern struct lxdream_config_entry alsa_config[];
bhaal22@643
    32
nkeynes@450
    33
gboolean lxdream_load_config_file( const gchar *filename );
nkeynes@450
    34
gboolean lxdream_save_config_file( const gchar *filename );
nkeynes@450
    35
gboolean lxdream_load_config_stream( FILE *f );
nkeynes@450
    36
gboolean lxdream_save_config_stream( FILE *f );
nkeynes@450
    37
nkeynes@450
    38
static struct lxdream_config_entry global_config[] =
nkeynes@736
    39
       {{ "bios", N_("Bios ROM"), CONFIG_TYPE_FILE, "dcboot.rom" },
nkeynes@736
    40
        { "flash", N_("Flash ROM"), CONFIG_TYPE_FILE, "dcflash.rom" },
nkeynes@736
    41
        { "default path", N_("Default disc path"), CONFIG_TYPE_PATH, "." },
nkeynes@736
    42
        { "save path", N_("Save-state path"), CONFIG_TYPE_PATH, "save" },
nkeynes@736
    43
        { "bootstrap", N_("Bootstrap IP.BIN"), CONFIG_TYPE_FILE, "IP.BIN" },
nkeynes@736
    44
        { "gdrom", NULL, CONFIG_TYPE_FILE, NULL },
nkeynes@736
    45
        { "recent", NULL, CONFIG_TYPE_FILE, NULL },
nkeynes@736
    46
        { NULL, CONFIG_TYPE_NONE }};
nkeynes@450
    47
nkeynes@450
    48
static struct lxdream_config_entry serial_config[] =
nkeynes@736
    49
       {{ "device", N_("Serial device"), CONFIG_TYPE_FILE, "/dev/ttyS1" },
nkeynes@736
    50
        { NULL, CONFIG_TYPE_NONE }};
nkeynes@450
    51
nkeynes@450
    52
struct lxdream_config_group lxdream_config_root[] = 
nkeynes@736
    53
       {{ "global", global_config },
nkeynes@736
    54
        { "controllers", NULL },
nkeynes@736
    55
        { "serial", serial_config },
bhaal22@643
    56
#ifdef HAVE_ALSA
nkeynes@736
    57
        { "alsa", alsa_config },
bhaal22@643
    58
#endif
nkeynes@736
    59
        { NULL, CONFIG_TYPE_NONE }};
nkeynes@450
    60
nkeynes@450
    61
static gchar *lxdream_config_load_filename = NULL;
nkeynes@450
    62
static gchar *lxdream_config_save_filename = NULL;
nkeynes@450
    63
nkeynes@450
    64
gboolean lxdream_find_config()
nkeynes@450
    65
{
nkeynes@480
    66
    gboolean result = TRUE;
nkeynes@450
    67
    char *home = getenv("HOME");
nkeynes@450
    68
    if( lxdream_config_save_filename == NULL ) {
nkeynes@736
    69
        lxdream_config_save_filename = g_strdup_printf("%s/.%s", home, DEFAULT_CONFIG_FILENAME);
nkeynes@450
    70
    }
nkeynes@450
    71
    if( lxdream_config_load_filename == NULL ) {
nkeynes@866
    72
        char *sysconfig = g_strdup_printf("%s/%s", get_sysconf_path(), DEFAULT_CONFIG_FILENAME);
nkeynes@736
    73
        if( access(lxdream_config_save_filename, R_OK) == 0 ) {
nkeynes@736
    74
            lxdream_config_load_filename = g_strdup(lxdream_config_save_filename);
nkeynes@866
    75
            g_free(sysconfig);
nkeynes@866
    76
        } else if( access( sysconfig, R_OK ) == 0 ) {
nkeynes@866
    77
            lxdream_config_load_filename = sysconfig;
nkeynes@736
    78
        } else if( access( "./" DEFAULT_CONFIG_FILENAME, R_OK ) == 0 ) {
nkeynes@736
    79
            lxdream_config_load_filename = g_strdup("./" DEFAULT_CONFIG_FILENAME);
nkeynes@866
    80
            g_free(sysconfig);
nkeynes@736
    81
        } else {
nkeynes@736
    82
            lxdream_config_load_filename = g_strdup(lxdream_config_save_filename);
nkeynes@866
    83
            g_free(sysconfig);
nkeynes@736
    84
            result = FALSE;
nkeynes@736
    85
        }	
nkeynes@480
    86
    }
nkeynes@480
    87
    return result;
nkeynes@450
    88
}
nkeynes@450
    89
nkeynes@450
    90
void lxdream_set_config_filename( const gchar *filename )
nkeynes@450
    91
{
nkeynes@450
    92
    if( lxdream_config_load_filename != NULL ) {
nkeynes@736
    93
        g_free(lxdream_config_load_filename);
nkeynes@450
    94
    }
nkeynes@450
    95
    lxdream_config_load_filename = g_strdup(filename);
nkeynes@450
    96
    if( lxdream_config_save_filename != NULL ) {
nkeynes@736
    97
        g_free(lxdream_config_save_filename);
nkeynes@450
    98
    }
nkeynes@450
    99
    lxdream_config_save_filename = g_strdup(filename);
nkeynes@450
   100
}
nkeynes@450
   101
nkeynes@450
   102
void lxdream_set_default_config( )
nkeynes@450
   103
{
nkeynes@450
   104
    struct lxdream_config_group *group = lxdream_config_root;
nkeynes@450
   105
    while( group->key != NULL ) {
nkeynes@736
   106
        struct lxdream_config_entry *param = group->params;
nkeynes@736
   107
        if( param != NULL ) {
nkeynes@736
   108
            while( param->key != NULL ) {
nkeynes@736
   109
                if( param->value != param->default_value ) {
nkeynes@736
   110
                    if( param->value != NULL )
nkeynes@736
   111
                        free( param->value );
nkeynes@736
   112
                    param->value = (gchar *)param->default_value;
nkeynes@736
   113
                }
nkeynes@736
   114
                param++;
nkeynes@736
   115
            }
nkeynes@736
   116
        }
nkeynes@736
   117
        group++;
nkeynes@450
   118
    }
nkeynes@450
   119
    maple_detach_all();
nkeynes@450
   120
}
nkeynes@450
   121
nkeynes@450
   122
const gchar *lxdream_get_config_value( int key )
nkeynes@450
   123
{
nkeynes@450
   124
    return global_config[key].value;
nkeynes@450
   125
}
nkeynes@450
   126
nkeynes@460
   127
void lxdream_set_config_value( lxdream_config_entry_t param, const gchar *value )
nkeynes@450
   128
{
nkeynes@475
   129
    if( param->value != value ) {
nkeynes@736
   130
        if( param->value != param->default_value && param->value != NULL ) {
nkeynes@736
   131
            free( param->value );
nkeynes@736
   132
        }
nkeynes@736
   133
        param->value = g_strdup(value);
nkeynes@450
   134
    }
nkeynes@450
   135
}
nkeynes@450
   136
nkeynes@460
   137
void lxdream_set_global_config_value( int key, const gchar *value )
nkeynes@460
   138
{
nkeynes@460
   139
    lxdream_set_config_value(&global_config[key], value);
nkeynes@460
   140
}
nkeynes@460
   141
nkeynes@759
   142
const struct lxdream_config_entry * lxdream_get_config_entry( int key )
nkeynes@724
   143
{
nkeynes@724
   144
    return &global_config[key];
nkeynes@724
   145
}
nkeynes@724
   146
nkeynes@458
   147
gboolean lxdream_set_group_value( lxdream_config_group_t group, const gchar *key, const gchar *value )
nkeynes@458
   148
{
nkeynes@458
   149
    int i;
nkeynes@458
   150
    for( i=0; group->params[i].key != NULL; i++ ) {
nkeynes@736
   151
        if( strcasecmp( group->params[i].key, key ) == 0 ) {
nkeynes@736
   152
            lxdream_set_config_value( &group->params[i], value );
nkeynes@736
   153
            return TRUE;
nkeynes@736
   154
        }
nkeynes@458
   155
    }
nkeynes@458
   156
    return FALSE;
nkeynes@458
   157
}
nkeynes@458
   158
nkeynes@460
   159
void lxdream_copy_config_list( lxdream_config_entry_t dest, lxdream_config_entry_t src )
nkeynes@460
   160
{
nkeynes@460
   161
    int i;
nkeynes@460
   162
    for( i=0; src[i].key != NULL; i++ ) {
nkeynes@736
   163
        lxdream_set_config_value( &dest[i], src[i].value );
nkeynes@460
   164
    }
nkeynes@460
   165
}
nkeynes@460
   166
nkeynes@450
   167
gboolean lxdream_load_config( )
nkeynes@450
   168
{
nkeynes@450
   169
    if( lxdream_config_load_filename == NULL ) {
nkeynes@736
   170
        lxdream_find_config();
nkeynes@450
   171
    }
nkeynes@450
   172
    return lxdream_load_config_file(lxdream_config_load_filename);
nkeynes@450
   173
}
nkeynes@450
   174
nkeynes@450
   175
gboolean lxdream_save_config( )
nkeynes@450
   176
{
nkeynes@450
   177
    if( lxdream_config_save_filename == NULL ) {
nkeynes@736
   178
        lxdream_find_config();
nkeynes@450
   179
    }
nkeynes@450
   180
    return lxdream_save_config_file(lxdream_config_save_filename);
nkeynes@450
   181
}
nkeynes@450
   182
nkeynes@450
   183
gboolean lxdream_load_config_file( const gchar *filename )
nkeynes@450
   184
{
nkeynes@450
   185
    FILE *f;
nkeynes@450
   186
    gboolean result;
nkeynes@450
   187
nkeynes@450
   188
    if( access(filename, F_OK) != 0 ) {
nkeynes@736
   189
        INFO( "Configuration file '%s' does not exist, creating from defaults" );
nkeynes@736
   190
        lxdream_set_default_config();
nkeynes@736
   191
        lxdream_save_config();
nkeynes@450
   192
    }
nkeynes@450
   193
nkeynes@450
   194
    f = fopen(filename, "ro");
nkeynes@450
   195
    if( f == NULL ) {
nkeynes@736
   196
        ERROR( "Unable to open configuration file '%s': %s", filename, strerror(errno) );
nkeynes@736
   197
        lxdream_set_default_config();
nkeynes@736
   198
        return FALSE;
nkeynes@450
   199
    }
nkeynes@450
   200
nkeynes@450
   201
    result = lxdream_load_config_stream( f );
nkeynes@450
   202
    fclose(f);
nkeynes@450
   203
    return result;
nkeynes@450
   204
}
nkeynes@450
   205
nkeynes@450
   206
gboolean lxdream_load_config_stream( FILE *f )
nkeynes@450
   207
{
nkeynes@450
   208
nkeynes@450
   209
    char buf[512];
nkeynes@450
   210
    int maple_device = -1, maple_subdevice = -1;
nkeynes@450
   211
    struct lxdream_config_group devgroup;
nkeynes@450
   212
    struct lxdream_config_group *group = NULL;
nkeynes@450
   213
    maple_device_t device = NULL;
nkeynes@450
   214
    lxdream_set_default_config();
nkeynes@450
   215
nkeynes@450
   216
    while( fgets( buf, sizeof(buf), f ) != NULL ) {
nkeynes@736
   217
        g_strstrip(buf);
nkeynes@736
   218
        if( buf[0] == '#' )
nkeynes@736
   219
            continue;
nkeynes@736
   220
        if( *buf == '[' ) {
nkeynes@736
   221
            char *p = strchr(buf, ']');
nkeynes@736
   222
            if( p != NULL ) {
nkeynes@736
   223
                struct lxdream_config_group *tmp_group;
nkeynes@736
   224
                maple_device = maple_subdevice = -1;
nkeynes@736
   225
                *p = '\0';
nkeynes@736
   226
                g_strstrip(buf+1);
nkeynes@736
   227
                tmp_group = &lxdream_config_root[0];
nkeynes@736
   228
                while( tmp_group->key != NULL ) {
nkeynes@736
   229
                    if( strcasecmp(tmp_group->key, buf+1) == 0 ) {
nkeynes@736
   230
                        group = tmp_group;
nkeynes@736
   231
                        break;
nkeynes@736
   232
                    }
nkeynes@736
   233
                    tmp_group++;
nkeynes@736
   234
                }
nkeynes@736
   235
            }
nkeynes@736
   236
        } else if( group != NULL ) {
nkeynes@736
   237
            char *value = strchr( buf, '=' );
nkeynes@736
   238
            if( value != NULL ) {
nkeynes@736
   239
                struct lxdream_config_entry *param = group->params;
nkeynes@736
   240
                *value = '\0';
nkeynes@736
   241
                value++;
nkeynes@736
   242
                g_strstrip(buf);
nkeynes@736
   243
                g_strstrip(value);
nkeynes@736
   244
                if( strcmp(group->key,"controllers") == 0  ) {
nkeynes@736
   245
                    if( g_strncasecmp( buf, "device ", 7 ) == 0 ) {
nkeynes@736
   246
                        maple_device = strtoul( buf+7, NULL, 0 );
nkeynes@736
   247
                        if( maple_device < 0 || maple_device > 3 ) {
nkeynes@736
   248
                            ERROR( "Device number must be between 0..3 (not '%s')", buf+7);
nkeynes@736
   249
                            continue;
nkeynes@736
   250
                        }
nkeynes@736
   251
                        maple_subdevice = 0;
nkeynes@736
   252
                        device = maple_new_device( value );
nkeynes@736
   253
                        if( device == NULL ) {
nkeynes@736
   254
                            ERROR( "Unrecognized device '%s'", value );
nkeynes@736
   255
                        } else {
nkeynes@736
   256
                            devgroup.key = "controllers";
nkeynes@736
   257
                            devgroup.params = maple_get_device_config(device);
nkeynes@736
   258
                            maple_attach_device( device, maple_device, maple_subdevice );
nkeynes@736
   259
                            group = &devgroup;
nkeynes@736
   260
                        }
nkeynes@736
   261
                        continue;
nkeynes@736
   262
                    } else if( g_strncasecmp( buf, "subdevice ", 10 ) == 0 ) {
nkeynes@736
   263
                        maple_subdevice = strtoul( buf+10, NULL, 0 );
nkeynes@736
   264
                        if( maple_device == -1 ) {
nkeynes@736
   265
                            ERROR( "Subdevice not allowed without primary device" );
nkeynes@736
   266
                        } else if( maple_subdevice < 1 || maple_subdevice > 5 ) {
nkeynes@736
   267
                            ERROR( "Subdevice must be between 1..5 (not '%s')", buf+10 );
nkeynes@736
   268
                        } else if( (device = maple_new_device(value)) == NULL ) {
nkeynes@736
   269
                            ERROR( "Unrecognized subdevice '%s'", value );
nkeynes@736
   270
                        } else {
nkeynes@736
   271
                            devgroup.key = "controllers";
nkeynes@736
   272
                            devgroup.params = maple_get_device_config(device);
nkeynes@736
   273
                            maple_attach_device( device, maple_device, maple_subdevice );
nkeynes@736
   274
                            group = &devgroup;
nkeynes@736
   275
                        }
nkeynes@736
   276
                        continue;
nkeynes@736
   277
                    }
nkeynes@736
   278
                }
nkeynes@736
   279
                while( param->key != NULL ) {
nkeynes@736
   280
                    if( strcasecmp( param->key, buf ) == 0 ) {
nkeynes@736
   281
                        param->value = g_strdup(value);
nkeynes@736
   282
                        break;
nkeynes@736
   283
                    }
nkeynes@736
   284
                    param++;
nkeynes@736
   285
                }
nkeynes@736
   286
            }
nkeynes@736
   287
        }
nkeynes@450
   288
    }
nkeynes@450
   289
    return TRUE;
nkeynes@450
   290
}
nkeynes@450
   291
nkeynes@450
   292
gboolean lxdream_save_config_file( const gchar *filename )
nkeynes@450
   293
{
nkeynes@450
   294
    FILE *f = fopen(filename, "wo");
nkeynes@450
   295
    gboolean result;
nkeynes@450
   296
    if( f == NULL ) {
nkeynes@736
   297
        ERROR( "Unable to open '%s': %s", filename, strerror(errno) );
nkeynes@736
   298
        return FALSE;
nkeynes@450
   299
    }
nkeynes@450
   300
    result = lxdream_save_config_stream(f);
nkeynes@450
   301
    fclose(f);
nkeynes@450
   302
    return TRUE;
nkeynes@450
   303
}    
nkeynes@450
   304
nkeynes@450
   305
gboolean lxdream_save_config_stream( FILE *f )
nkeynes@450
   306
{
nkeynes@450
   307
    struct lxdream_config_group *group = &lxdream_config_root[0];
nkeynes@736
   308
nkeynes@450
   309
    while( group->key != NULL ) {
nkeynes@736
   310
        struct lxdream_config_entry *entry = group->params;
nkeynes@736
   311
        fprintf( f, "[%s]\n", group->key );
nkeynes@736
   312
nkeynes@736
   313
        if( entry != NULL ) {
nkeynes@736
   314
            while( entry->key != NULL ) {
nkeynes@736
   315
                if( entry->value != NULL ) {
nkeynes@736
   316
                    fprintf( f, "%s = %s\n", entry->key, entry->value );
nkeynes@736
   317
                }
nkeynes@736
   318
                entry++;
nkeynes@736
   319
            }
nkeynes@736
   320
        } else if( strcmp(group->key, "controllers") == 0 ) {
nkeynes@736
   321
            int i,j;
nkeynes@736
   322
            for( i=0; i<4; i++ ) {
nkeynes@736
   323
                for( j=0; j<6; j++ ) {
nkeynes@736
   324
                    maple_device_t dev = maple_get_device( i, j );
nkeynes@736
   325
                    if( dev != NULL ) {
nkeynes@736
   326
                        if( j == 0 )
nkeynes@736
   327
                            fprintf( f, "Device %d = %s\n", i, dev->device_class->name );
nkeynes@736
   328
                        else 
nkeynes@736
   329
                            fprintf( f, "Subdevice %d = %s\n", j, dev->device_class->name );
nkeynes@736
   330
                        if( dev->get_config != NULL && ((entry = dev->get_config(dev)) != NULL) ) {
nkeynes@736
   331
                            while( entry->key != NULL ) {
nkeynes@736
   332
                                if( entry->value != NULL ) {
nkeynes@736
   333
                                    fprintf( f, "%*c%s = %s\n", j==0?4:8, ' ',entry->key, entry->value );
nkeynes@736
   334
                                }
nkeynes@736
   335
                                entry++;
nkeynes@736
   336
                            }
nkeynes@736
   337
                        }
nkeynes@736
   338
                    }
nkeynes@736
   339
                }
nkeynes@736
   340
            }
nkeynes@736
   341
        }
nkeynes@736
   342
        fprintf( f, "\n" );
nkeynes@736
   343
        group++;
nkeynes@450
   344
    }
nkeynes@450
   345
    return TRUE;
nkeynes@450
   346
}
.