all repos — mgba @ 4c38f769565e8ddd7d3a8eef1a41975206c129a0

mGBA Game Boy Advance Emulator

src/third-party/inih/README.md (view raw)

  1**inih (INI Not Invented Here)** is a simple [.INI file](http://en.wikipedia.org/wiki/INI_file) parser written in C. It's only a couple of pages of code, and it was designed to be _small and simple_, so it's good for embedded systems. It's also more or less compatible with Python's [ConfigParser](http://docs.python.org/library/configparser.html) style of .INI files, including RFC 822-style multi-line syntax and `name: value` entries.
  2
  3To use it, just give `ini_parse()` an INI file, and it will call a callback for every `name=value` pair parsed, giving you strings for the section, name, and value. It's done this way ("SAX style") because it works well on low-memory embedded systems, but also because it makes for a KISS implementation.
  4
  5You can also call `ini_parse_file()` to parse directly from a `FILE*` object, or `ini_parse_stream()` to parse using a custom reader to implement string-based or other custom I/O ([see example code](https://github.com/benhoyt/inih/blob/master/examples/ini_buffer.c)).
  6
  7Download a release, browse the source, or read about [how to use inih in a DRY style](http://blog.brush.co.nz/2009/08/xmacros/) with X-Macros.
  8
  9
 10## Compile-time options ##
 11
 12  * **Multi-line entries:** By default, inih supports multi-line entries in the style of Python's ConfigParser. To disable, add `-DINI_ALLOW_MULTILINE=0`.
 13  * **UTF-8 BOM:** By default, inih allows a UTF-8 BOM sequence (0xEF 0xBB 0xBF) at the start of INI files. To disable, add `-DINI_ALLOW_BOM=0`.
 14  * **Stack vs heap:** By default, inih allocates its line buffer on the stack. To allocate on the heap using `malloc` instead, specify `-DINI_USE_STACK=0`.
 15  * **Stop on first error:** By default, inih keeps parsing the rest of the file after an error. To stop parsing on the first error, add `-DINI_STOP_ON_FIRST_ERROR=1`.
 16  * **Maximum line length:** The default maximum line length is 200 bytes. To override this, add something like `-DINI_MAX_LINE=1000`.
 17
 18
 19## Simple example in C ##
 20
 21```c
 22#include <stdio.h>
 23#include <stdlib.h>
 24#include <string.h>
 25#include "../ini.h"
 26
 27typedef struct
 28{
 29    int version;
 30    const char* name;
 31    const char* email;
 32} configuration;
 33
 34static int handler(void* user, const char* section, const char* name,
 35                   const char* value)
 36{
 37    configuration* pconfig = (configuration*)user;
 38
 39    #define MATCH(s, n) strcmp(section, s) == 0 && strcmp(name, n) == 0
 40    if (MATCH("protocol", "version")) {
 41        pconfig->version = atoi(value);
 42    } else if (MATCH("user", "name")) {
 43        pconfig->name = strdup(value);
 44    } else if (MATCH("user", "email")) {
 45        pconfig->email = strdup(value);
 46    } else {
 47        return 0;  /* unknown section/name, error */
 48    }
 49    return 1;
 50}
 51
 52int main(int argc, char* argv[])
 53{
 54    configuration config;
 55
 56    if (ini_parse("test.ini", handler, &config) < 0) {
 57        printf("Can't load 'test.ini'\n");
 58        return 1;
 59    }
 60    printf("Config loaded from 'test.ini': version=%d, name=%s, email=%s\n",
 61        config.version, config.name, config.email);
 62    return 0;
 63}
 64```
 65
 66
 67## C++ example ##
 68
 69If you're into C++ and the STL, there is also an easy-to-use [INIReader class](https://github.com/benhoyt/inih/blob/master/cpp/INIReader.h) that stores values in a `map` and lets you `Get()` them:
 70
 71```cpp
 72#include <iostream>
 73#include "INIReader.h"
 74
 75int main()
 76{
 77    INIReader reader("../examples/test.ini");
 78
 79    if (reader.ParseError() < 0) {
 80        std::cout << "Can't load 'test.ini'\n";
 81        return 1;
 82    }
 83    std::cout << "Config loaded from 'test.ini': version="
 84              << reader.GetInteger("protocol", "version", -1) << ", name="
 85              << reader.Get("user", "name", "UNKNOWN") << ", email="
 86              << reader.Get("user", "email", "UNKNOWN") << ", pi="
 87              << reader.GetReal("user", "pi", -1) << ", active="
 88              << reader.GetBoolean("user", "active", true) << "\n";
 89    return 0;
 90}
 91```
 92
 93This simple C++ API works fine, but it's not very fully-fledged. I'm not planning to work more on the C++ API at the moment, so if you want a bit more power (for example `GetSections()` and `GetFields()` functions), see these forks:
 94
 95  * https://github.com/Blandinium/inih
 96  * https://github.com/OSSystems/inih
 97
 98
 99## Differences from ConfigParser ##
100
101Some differences between inih and Python's [ConfigParser](http://docs.python.org/library/configparser.html) standard library module:
102
103* INI name=value pairs given above any section headers are treated as valid items with no section (section name is an empty string). In ConfigParser having no section is an error.
104* Line continuations are handled with leading whitespace on continued lines (like ConfigParser). However, instead of concatenating continued lines together, they are treated as separate values for the same key (unlike ConfigParser).