mirror of
git://git.savannah.nongnu.org/eliot.git
synced 2025-01-13 20:03:23 +01:00
e7a8d01a8d
There are too many change to list properly, here is an overview of the main changes: - the dictionary is now in C++ - the dictionary has a new format, where it is possible to specify the letters, their points, their frequency, ... It is backwards compatible. - Eliot now supports non-ASCII characters everywhere - i18n of the compdic, listdic, regexpmain binaries - i18n of the wxWidgets interface (now in english by default)
207 lines
7.1 KiB
Text
207 lines
7.1 KiB
Text
|
|
The dictionary is a directed tree. Compression is achieved by
|
|
sharing word prefix and suffix. Search is NOT case sensitive.
|
|
|
|
considering this 3 words dictionary:
|
|
|
|
ABC
|
|
ADA
|
|
EDAA
|
|
|
|
The tree will look like this:
|
|
|
|
root
|
|
|
|
|
A----------E!
|
|
| |
|
|
B----D! D!
|
|
| | |
|
|
C!* | A!
|
|
| | |
|
|
| ------A!*
|
|
| |
|
|
| |
|
|
------------------- 0!* (sink)
|
|
|
|
The tree is saved using an array of 32 bits words.
|
|
A cell is a binary structure
|
|
|
|
ptr : index in the array of the first child
|
|
term : is it the last letter of a word (*)
|
|
last : is it the last child of its local root (!)
|
|
chr : guess what !
|
|
|
|
There is no pointer from a cell to its brother, it is simply the
|
|
next cell in the array (you know you are on the last brother when
|
|
the flag "last" is set).
|
|
|
|
The way it is stored in a file is a different thing! The tree is
|
|
stored bottom-up. The sink (offset 0) is the first cell of
|
|
the array.
|
|
|
|
Using compdic (which you can find in the eliot/dic directory),
|
|
the compiled dictionary will look like this:
|
|
|
|
compdic console output (cut in the middle):
|
|
===================================================================
|
|
dictionary name: ODS 4.0
|
|
compressed on: mer 12 déc 2007 07:29:50 GMT
|
|
compressed using a binary compiled by: ipkiss@ulukai
|
|
dictionary type: DAWG
|
|
letters: ABCDEFGHIJKLMNOPQRSTUVWXYZ?
|
|
number of letters: 27
|
|
number of words: 369085
|
|
header size: 360 bytes
|
|
root: 100950 (edge)
|
|
nodes: 40377 used + 418387 saved
|
|
edges: 100950 used + 601922 saved
|
|
===============================================
|
|
letter | points | frequency | vowel | consonant
|
|
-------+--------+-----------+-------+----------
|
|
A | 1 | 9 | 1 | 0
|
|
B | 3 | 2 | 0 | 1
|
|
C | 3 | 2 | 0 | 1
|
|
D | 2 | 3 | 0 | 1
|
|
[... output cut here ...]
|
|
X | 10 | 1 | 0 | 1
|
|
Y | 10 | 1 | 1 | 1
|
|
Z | 10 | 1 | 0 | 1
|
|
? | 0 | 2 | 1 | 1
|
|
===============================================
|
|
Load time: 0,060 s
|
|
Compression time: 0,170 s
|
|
Maximum recursion level reached: 16
|
|
===================================================================
|
|
|
|
binary view of the dictionary (FIXME: not up to date):
|
|
===================================================================
|
|
0001 0203 0405 0607 0809 0a0b 0c0d 0e0f
|
|
00000000: 5f43 4f4d 5049 4c45 445f 4449 4354 494f _COMPILED_DICTIO
|
|
00000010: 4e41 5259 5f00 0000 0900 0000 0300 0000 NARY_...........
|
|
00000020: 0900 0000 0700 0000 0100 0000 0100 0000 ................
|
|
00000030: 0000 0002 0000 001b 0000 000b 0100 0010 ................
|
|
00000040: 0200 0022 0200 000a 0500 0022 0300 0008 ..."......."....
|
|
00000050: 0600 002a 0700 0000 ...*....
|
|
===================================================================
|
|
|
|
The header is made of 2 structures (for backwards compatibility
|
|
with older headers) like this:
|
|
===================================================================
|
|
#define _COMPIL_KEYWORD_ "_COMPILED_DICTIONARY_"
|
|
|
|
struct Dict_header_old // offset
|
|
{
|
|
char ident[sizeof(_COMPIL_KEYWORD_)]; // 0x00
|
|
uint8_t version; // 0x16
|
|
char unused; // 0x17
|
|
uint32_t root; // 0x18
|
|
uint32_t nwords; // 0x1c
|
|
uint32_t edgesused; // 0x20
|
|
uint32_t nodesused; // 0x24
|
|
uint32_t nodessaved; // 0x28
|
|
uint32_t edgessaved; // 0x2c
|
|
};
|
|
|
|
#define _MAX_USER_HOST_ 32
|
|
#define _MAX_DIC_NAME_SIZE_ 30
|
|
#define _MAX_LETTERS_NB_ 63
|
|
#define _MAX_LETTERS_SIZE_ 80
|
|
|
|
struct Dict_header
|
|
{
|
|
uint64_t compressDate;
|
|
// Build information
|
|
char userHost[_MAX_USER_HOST_];
|
|
// Size taken by the build information
|
|
uint32_t userHostSize;
|
|
|
|
// Compression algorithm (1 = DAWG, 2 = GADDAG)
|
|
uint8_t algorithm;
|
|
// Variant used in the rules (XXX: currently unused)
|
|
uint8_t variant;
|
|
|
|
// Dictionary official name and version (e.g.: ODS 5.0)
|
|
char dicName[_MAX_DIC_NAME_SIZE_];
|
|
// Size taken by the dictionary name
|
|
uint32_t dicNameSize;
|
|
|
|
// Letters used in the dictionary
|
|
// We should have: nbLetters <= lettersSize <= _MAX_LETTERS_SIZE_
|
|
// and: nbLetters <= _MAX_LETTERS_NB_
|
|
// The letters themselves, in UTF-8
|
|
char letters[_MAX_LETTERS_SIZE_];
|
|
// Size taken by the letters
|
|
uint32_t lettersSize;
|
|
// Number of letters (XXX: in theory useless, but allows a sanity check)
|
|
uint32_t nbLetters;
|
|
|
|
// Points of the letters (indexed by their code)
|
|
// The "+ 1" is there for struct alignment
|
|
uint8_t points[_MAX_LETTERS_NB_ + 1];
|
|
// Frequency of the letters (indexedy their code)
|
|
// The "+ 1" is there for struct alignment
|
|
uint8_t frequency[_MAX_LETTERS_NB_ + 1];
|
|
// Bitfield indicating whether letters are vowels
|
|
uint64_t vowels;
|
|
// Bitfield indicating whether letters are consonants
|
|
uint64_t consonants;
|
|
}
|
|
===================================================================
|
|
|
|
In the old version of the dictionary, only the first structure was used
|
|
(with version = 0). The current format (version = 1) has the 2 structs
|
|
next to each other.
|
|
The dictionary name, the letters, and the user/host information are
|
|
stored in UTF-8. All the numbers are big endian (i.e. the output of
|
|
the htonl() function).
|
|
To avoid alignment issues, the extended header has been designed to
|
|
have multiples of 64 bits regularly.
|
|
|
|
|
|
binary output of the header (FIXME: not up to date):
|
|
===================================================================
|
|
0x00 ident : _COMPILED_DICTIONARY_
|
|
0x16 version : 0 00000001
|
|
0x17 unused : 0 00000000
|
|
0x18 root : 9 00000009
|
|
0x1c words : 3 00000003
|
|
0x20 edges used : 9 00000009
|
|
0x24 nodes used : 7 00000007
|
|
0x28 nodes saved : 1 00000001
|
|
0x2c edges saved : 1 00000001
|
|
===================================================================
|
|
|
|
The real array of data begins at offset 0x168. The array is stored
|
|
'as is' right after the header. Each array cell is a bit-structure
|
|
on 4 bytes:
|
|
|
|
struct DicEdge
|
|
{
|
|
unsigned int ptr : 24;
|
|
unsigned int term : 1;
|
|
unsigned int last : 1;
|
|
unsigned int chr : 6;
|
|
};
|
|
|
|
Characters are not stored in ASCII. The order of the letters given
|
|
to the compdic binary is preserved, but we changed the values: the
|
|
first letter is 1, the second one is 2, etc...
|
|
The dictionary can thus handle up to 64 different letters but not more.
|
|
The letter 0 is special (used for the sink node in particular), so
|
|
in practice there are only 63 distinct letters.
|
|
|
|
offs binary structure
|
|
---- -------- | ------------------
|
|
0x00 02000000 | 0 ptr= 0 t=0 l=1 chr=0 (`)
|
|
0x04 1b000000 | 1 ptr= 0 t=1 l=1 chr=3 (c)
|
|
0x08 0b000000 | 2 ptr= 0 t=1 l=1 chr=1 (a)
|
|
0x0c 10000001 | 3 ptr= 1 t=0 l=0 chr=2 (b)
|
|
0x10 22000002 | 4 ptr= 2 t=0 l=1 chr=4 (d)
|
|
0x14 0a000002 | 5 ptr= 2 t=0 l=1 chr=1 (a)
|
|
0x18 22000005 | 6 ptr= 5 t=0 l=1 chr=4 (d)
|
|
0x1c 08000003 | 7 ptr= 3 t=0 l=0 chr=1 (a)
|
|
0x20 2a000006 | 8 ptr= 6 t=0 l=1 chr=5 (e)
|
|
0x24 00000007 | 9 ptr= 7 t=0 l=0 chr=0 (`)
|
|
|
|
Strictly speaking, there is no node in the graph, only labelled edges.
|
|
|