File: aspell-dev.info, Node: Top, Next: Style Guidelines, Prev: (dir), Up: (dir) Notes ***** This manual is designed for those who wish to develop Aspell. It is currently very sketchy. However, it should improve over time. The latest version of this document can be found at `http://savannah.gnu.org/download/aspell/manual/devel/devel.html'. * Menu: * Style Guidelines:: * How to Submit a Patch:: * C++ Standard Library:: * Templates:: * Error Handling:: * Source Code Layout :: * Strings:: * Smart Pointers:: * I/O:: * Config Class:: * Filter Interface:: * Filter Modes:: * Data Structures:: * Mk-Src Script:: * How It All Works:: * Copying:: File: aspell-dev.info, Node: Style Guidelines, Next: How to Submit a Patch, Prev: Top, Up: Top 1 Style Guidelines ****************** * Style Guidelines:: * How to Submit a Patch:: * C++ Standard Library:: * Templates:: * Error Handling:: * Source Code Layout :: * Strings:: * Smart Pointers:: * I/O:: * Config Class:: * Filter Interface:: * Filter Modes:: * Data Structures:: * Mk-Src Script:: * How It All Works:: * Copying:: As far as coding styles go I am really not that picky. The important thing is to stay consistent. However, please whatever you do, do not indent with more than 4 characters as I find indenting with more than that extremely difficult to read as most of the code ends up on the right side of the screen. File: aspell-dev.info, Node: How to Submit a Patch, Next: C++ Standard Library, Prev: Style Guidelines, Up: Top 2 How to Submit a Patch *********************** Bug reports and patches should be submitted via the Sourceforge Tracker at `http://sourceforge.net/tracker/?group_id=245' rather than being posted to any of the mailing lists. The mailing lists are good if you need to check something out or need help or feedback from other readers, but they are not the best place to submit bugs or patches because they will likely get forgotten or lost within the mailing list traffic if not acted upon immediately. Please make the effort to use the tracker. File: aspell-dev.info, Node: C++ Standard Library, Next: Templates, Prev: How to Submit a Patch, Up: Top 3 C++ Standard Library ********************** The C++ Standard library is not used directly except under very specific circumstances. The string class and the STL are used indirectly through wrapper classes and all I/O is done using the standard C library with light right helper classes to make using C I/O a bit more C++ like. However the `new', `new[]', `delete' and `delete[]' operators are used to allocate memory when appropriate. File: aspell-dev.info, Node: Templates, Next: Error Handling, Prev: C++ Standard Library, Up: Top 4 Templates *********** Templates are used in Aspell when there is a clear advantage to doing so. Whenever you use templates please use them carefully and try very hard not to create code bloat by generating a lot of unnecessary and duplicate code. File: aspell-dev.info, Node: Error Handling, Next: Source Code Layout, Prev: Templates, Up: Top 5 Error Handling **************** Exceptions are not used in Aspell as I find them more trouble than they are worth. Instead an alternate method of error handling is used which is based around the PosibErr class. PosibErr is a special Error handling device that will make sure that an error is properly handled. It is defined in `posib_err.hpp'. PosibErr is expected to be used as the return type of the function. It will automatically be converted to the "normal" return type however if the normal returned type is accessed and there is an "unhandled" error condition it will abort. It will also abort if the object is destroyed with an "unhandled" error condition. This includes ignoring the return type of a function returning an error condition. An error condition is handled by simply checking for the presence of an error, calling ignore, or taking ownership of the error. The PosibErr class is used extensively throughout Aspell. Please refer to the Aspell source for examples of using PosibErr until better documentation is written. File: aspell-dev.info, Node: Source Code Layout, Next: Strings, Prev: Error Handling, Up: Top 6 Source Code Layout ******************** `common/' Common code used by all parts of Aspell. `lib/' Library code used only by the actual Aspell library. `data/' Data files used by Aspell. `modules/' Aspell modules which are eventually meant to be pluggable. `speller/' `default/' Main speller Module. `filter/' `tokenizer/' `auto/' Scripts and data files to automatically generate code used by Aspell. `interface/' Header files and such that external programs should use when in order to use the Aspell library. `cc/' The external C interface that programs should be using when they wish to use Aspell. `prog/' Actual programs based on the Aspell library. The main Aspell utility is included here. `scripts/' Miscellaneous scripts used by Aspell. `manual/' `examples/' Example programs demonstrating the use of the Aspell library. File: aspell-dev.info, Node: Strings, Next: Smart Pointers, Prev: Source Code Layout, Up: Top 7 Strings ********* 7.1 String ========== The `String' class provided the same functionality of the C++ string except for fewer constructors. It also inherits `OStream' so that you can write to it with the `<<' operator. It is defined in `string.hpp'. 7.2 ParmString ============== ParmString is a special string class that is designed to be used as a parameter for a function that is expecting a string. It is defined in `parm_string.hpp'. It will allow either a `const char *' or `String' class to be passed in. It will automatically convert to a `const char *'. The string can also be accessed via the `str' method. Usage example: void foo(ParmString s1, ParmString s2) { const char * str0 = s1; unsigned int size0 = s2.size() if (s1 == s2 || s2 == "bar") { ... } } ... String s1 = "..."; foo(s1); const char * s2 = "..."; foo(s2); This class should be used when a string is being passed in as a parameter. It is faster than using `const String &' (as that will create an unnecessary temporary when a `const char *' is passed in), and is less annoying than using `const char *' (as it doesn't require the `c_str()' method to be used when a `String' is passed in). 7.3 CharVector ============== A character vector is basically a `Vector<char>' but it has a few additional methods for dealing with strings which `Vector' does not provide. It, like `String', is also inherits `OStream' so that you can write to it with the `<<' operator. It is defined in `char_vector.hpp'. Use it when ever you need a string which is guaranteed to be in a continuous block of memory which you can write to. File: aspell-dev.info, Node: Smart Pointers, Next: I/O, Prev: Strings, Up: Top 8 Smart Pointers **************** Smart pointers are used extensively in Aspell to simplify memory management tasks and to avoid memory leaks. 8.1 CopyPtr =========== The `CopyPtr' class makes a deep copy of an object whenever it is copied. The `CopyPtr' class is defined in `copy_ptr.hpp'. This header should be included wherever `CopyPtr' is used. The complete definition of the object `CopyPtr' is pointing to does not need to be defined at this point. The implementation is defined in `copy_ptr-t.hpp'. The implementation header file should be included at a point in your code where the class `CopyPtr' is pointing to is completely defined. 8.2 ClonePtr ============ `ClonePtr' is like copy pointer except the `clone()' method is used instead of the copy constructor to make copies of an object. If is defined in `clone_ptr.hpp' and implemented in `clone_ptr-t.hpp'. 8.3 StackPtr ============ A `StackPtr' is designed to be used whenever the only pointer to a new object allocated with `new' is on the stack. It is similar to the standard C++ `auto_ptr' but the semantics are a bit different. It is defined in `stack_ptr.hpp' -- unlike `CopyPtr' or `ClonePtr' it is defined and implemented in this header file. 8.4 GenericCopyPtr ================== A generalized version of `CopyPtr' and `ClonePtr' which the two are based on. It is defined in `generic_copy_ptr.hpp' and implemented in `generic_copy_ptr-t.hpp'. File: aspell-dev.info, Node: I/O, Next: Config Class, Prev: Smart Pointers, Up: Top 9 I/O ***** Aspell does not use C++ I/O classes and functions in any way since they do not provide a way to get at the underlying file number and can often be slower than the highly tuned C I/O functions found in the standard C library. However, some lightweight wrapper classes are provided so that standard C I/O can be used in a more C++ like way. 9.1 IStream/OStream =================== These two base classes mimic some of the functionally of the C++ functionally of the corresponding classes. They are defined in `istream.hpp' and `ostream.hpp' respectively. They are however based on standard C I/O and are not proper C++ streams. 9.2 FStream =========== Defined in `fstream.hpp'. 9.3 Standard Streams ==================== `CIN'/`COUT'/`CERR'. Defined in `iostream.hpp'. File: aspell-dev.info, Node: Config Class, Next: Filter Interface, Prev: I/O, Up: Top 10 Config Class *************** The `Config' class is used to hold configuration information. It has a set of keys which it will accept. Inserting or even trying to look at a key that it does not know will produce an error. It is defined in `common/config.hpp'. File: aspell-dev.info, Node: Filter Interface, Next: Filter Modes, Prev: Config Class, Up: Top 11 Filter Interface ******************* 11.1 Overview ============= In Aspell there are 5 types of filters: 1. _Decoders_ which take input in some standard format such as iso8859-1 or UTF-8 and convert it into a string of `FilterChars'. 2. _Decoding filters_ which manipulate a string of `FilterChars' by decoding the text is some way such as converting an SGML character into its Unicode value. 3. _True filters_ which manipulate a string of `FilterChars' to make it more suitable for spell checking. These filters generally blank out text which should not be spell checked 4. _Encoding filters_ which manipulate a string of `FilterChars' by encoding the text in some way such as converting certain Unicode characters to SGML characters. 5. _Encoders_ which take a string of `FilterChars' and convert into a standard format such as iso8859-1 or UTF-8 Which types of filters are used depends on the situation 1. When _decoding words_ for spell checking: * The _decoder_ to convert from a standard format * The _decoding filter_ to perform high level decoding if necessary * The _encoder_ to convert into an internal format used by the speller module 2. When _checking a document_ * The _decoder_ to convert from a standard format * The _decoding filter_ to perform high level decoding if necessary * A _true filter_ to filter out parts of the document which should not be spell checked * The _encoder_ to convert into an internal format used by the speller module 3. When _encoding words_ such as those returned for suggestions: * The _decoder_ to convert from the internal format used by the speller module * The _encoding filter_ to perform high level encodings if necessary * The _encoder_ to convert into a standard format A `FilterChar' is a struct defined in `common/filter_char.hpp' which contains two members, a character, and a width. Its purpose is to keep track of the width of the character in the original format. This is important because when a misspelled word is found the exact location of the word needs to be returned to the application so that it can highlight it for the user. For example if the filters translated this: Mr. foo said "I hate my namme". to this Mr. foo said "I hate my namme". without keeping track of the original width of the characters the application will likely highlight `e my ' as the misspelling because the spell checker will return 25 as the offset instead of 30. However with keeping track of the width using `FilterChar' the spell checker will know that the real position is 30 since the quote is really 6 characters wide. In particular the text will be annotated something like the following: 1111111111111611111111111111161 Mr. foo said "I hate my namme". The standard _encoder_ and _decoder_ filters are defined in `common/convert.cpp'. There should generally not be any need to deal with them so they will not be discussed here. The other three filters, the _encoding filter_, the _true filter_, and the _decoding filter_, are all defined the exact same way; they are inherited from the `IndividualFilter' class. 11.2 Adding a New Filter ======================== A new filter basically is added by placing the corresponding loadable object inside a directory reachable by Aspell via `filter-path' list. Further it is necessary that the corresponding filter description file is located in one of the directories listed by the `option-path' list. The name of the loadable object has to conform to the following convention `libfiltername-filter.so' where `filtername' stands for the name of the filter which is passed to Aspell by the `add-filter' option. The same applies to the filter description file which has to conform to the following naming scheme: `filtername-filter.opt'. To add a new loadable filter object create a new file. Basically the file should be a C++ file and end in `.cpp'. The file should contain a new filter class inherited from `IndividualFilter' and a constructor function called `new_filtertype' (see *Note Constructor Function::) returning a new filter object. Further it is necessary to manually generate the filter description file. Finally the resulting object has to be turned into a loadable filter object using libtool. Alternatively a new filter may extend the functionality of an existing filter. In this case the new filter has to be derived form the corresponding valid filter class instead of the `IndividualFilter' class. 11.3 IndividualFilter class =========================== All filters are required to inherit from the `IndividualFilter' class found in `indiv_filter.hpp'. See that file for more details and the other filter modules for examples of how it is used. 11.4 Constructor Function ========================= After the class is created a function must be created which will return a new filter allocated with `new'. The function must have the following prototype: C_EXPORT IndividualFilter * new_aspell_FILTERNAME_FILTERTYPE Filters are defined in groups where each group contains an _encoding filter_, a _true filter_, and a _decoding filter_ (see *Note Filter Overview::). Only one of them is required to be defined, however they all need a separate constructor function. 11.5 Filter Description File ============================ This file contains the description of a filter which is loaded by Aspell immediately when the `add-filter' option is invoked. If this file is missing Aspell will complain about it. It consists of lines containing comments which must be started by a `#' character and lines containing key value pairs describing the filter. Each file at least has to contain the following two lines in the given order. ASPELL >=0.60 DESCRIPTION this is short filter description The first non blank, non comment line has to contain the keyword `ASPELL' followed by the version of Aspell which the filter is usable with. To denote multiple Aspell versions the version number may be prefixed by ``<'', ``<='', ``='', ``>='' or ``>'. If the range prefix is omitted ``='' is assumed. The `DESCRIPTION' of the filter should be under 50, begin in lower case, and note include any trailing punctuation characters. The keyword `DESCRIPTION' may be abbreviated by `DESC'. For each filter feature (see *Note Filter Overview::) provided by the corresponding loadable object, the option file has to contain the following line: STATIC `filtertype' `filtertype' stands for one of `decoder', `filter' or `encoder' denoting the entire filter type. This line allows to statically (see *Note Link Filters Static::) link the filter into Aspell if requested by the user or by the system Aspell is built for. OPTION newoption DESCRIPTION this is a short description of newoption TYPE bool DEFAULT false ENDOPTION An option is added by a line containing the keyword `OPTION' followed by the name of the option. If this name is not prefixed by the name of the filter Aspell will implicitly do that. For the `DESCRIPTION' of a filter option the same holds as for the filter description. The `TYPE' of the option may be one of `bool', `int', `string' or `list'. If the `TYPE' is omitted `bool' is assumed. The default value(s) for an option is specified via `DEFAULT' (short `DEF') followed by the desired `TYPE' dependent default value. The table *Note Filter Default Values:: shows the possible values for each `TYPE'. Type Default Available bool true true false int 0 any number value string any printable string list any comma separated list of strings Table 1. Shows the default values Aspell assumes if option `description' lacks a `DEFAULT' or `DEF' line. The `ENDOPTION' line may be omitted as it is assumed implicitly if a line containing `OPTION', `STATIC'. *Note* The keywords in a filter description file are case insensitive. The above examples use the all uppercase for better distinguishing them from values and comments. Further a filter description may contain blank lines to enhance their readability. *Note* An option of `list' type may contain multiple consecutive lines for default values starting with `DEFAULT' or `DEF', to specify numerous default values. 11.6 Retrieve Options by a Filter ================================= An option always has to be retrieved by a filter using its full qualified name as the following example shows. config->retrieve_bool("filter-filtername-newoption"); The prefix `filter-' allows user to relate option uniquely to the specific filter when `filtername-newoption' ambiguous an existing option of Aspell. The `filtername' stands for the name of the filter the option belongs to and `-newoption' is the name of the option as specified in the corresponding `.opt' file (see *Note Filter Description File:: 11.7 Compiling and Linking ========================== See a good book on Unix programming on how to turn the filter source into a loadable object. 11.8 Programmer's Interface =========================== A more convenient way recommended, if filter is added to Aspell standard distribution to build a new filter is provided by Aspell's programmers interface for filter. It is provided by the `loadable-filter-API.hpp' file. Including this file gives access to a collection of macros hiding nasty details about runtime construction of a filter and about filter debugging. Table *Note Interface Macros:: shows the macros provided by the interface. For details upon the entire macros see `loadable-filter-API.hpp'. An example on how to use these macros can be found at `examples/loadable/ccpp-context.hpp' and `examples/loadable/ccpp-context.cpp'. Macro Type Description Notes ACTIVATE_ENCODER M makes the entire do not call inside encoding filter class declaration; callable by Aspell these macros define new_<filtertype> function; ACTIVATE_DECODER M makes the entire _as above_ decoding filter callable by Aspell ACTIVATE_FILTER M makes the entire filter _as above_ callable by Aspell FDEBUGOPEN D Initialises the macros These macros are only for debugging a filter active if the and opens the debug `FILTER_PROGRESS_CONTROL' file stream macro is defined and denotes the name of the file debug messages should be sent to. If debugging should go to Aspell standard debugging output (right now stderr) use empty string constant as filename FDEBUGNOTOPEN D Same as "FDEBUGOPEN" but _as above_ only if debug file stream was not opened yet FDEBUGCLOSE D closes the debugging _as above_ device opened by "FDEBUGOPEN" and reverts it to "stderr"; FDEBUG D prints the filename and _as above_ the line number it occurs FDEBUGPRINTF D special printf for _as above_ debugging Table 2. Shows the macros provided by `loadable-filter-API.hpp' (*M* mandatory, *D* debugging) 11.9 Adding a filter to Aspell standard distribution ==================================================== Any filter which one day should be added to Aspell has to be built using the developer interface, described in *Note Programmer's Interface::. To add the filter the following steps have to be performed: 1. Decide whether the filter should be kept loadable if possible, or always be statically linked to Aspell. 2. Place the filter sources inside the entire directory of Aspell source tree. Right now use `$top_srcdir/modules/filter'. 3. Modify the `Makefile.am' file on the topmost directory of the Aspell distribution. Follow the instructions given by the `#Filter Modules' section. 4. Run `autoconf', `automake', ... 5. Reconfigure sources. 6. Clear away any remains of a previous build and rebuild sources. 7. Reinstall Aspell. 8. Test if filter has been added properly otherwise return to steps 2-7 9. Reconfigure sources with `enable-static' flag and repeat steps 2-7 until your filter builds and runs properly in case of static linkage. 10. Add your source files to cvs, and commit all your changes. Or in case you are not allowed to commit to cvs submit a patch (see *Note How to Submit a Patch::) containing your changes. File: aspell-dev.info, Node: Filter Modes, Next: Data Structures, Prev: Filter Interface, Up: Top 12 Filter Modes *************** Filter modes are the preferred way to specify combinations of filters which are used regularly and thus abbreviate Aspell's command line arguments. A new filter mode is specified by a file named like the filter new mode and prefixed by `.amf' (Aspell Mode File). If such a file is accessible by the path set via filter-path option Aspell will try to load the contained mode specification. 12.1 Aspell Mode File ===================== The first key in the made file has be the `mode' key. It is checked against the mode name part of the .amf file. If the `mode' key is missing mode file will be rejected. The same holds for the `aspell' key which specifies the version(s) of Aspell which is(are) required by the filter. If these two keys are followed by at least one `magic' key Aspell will be able to select the entire mode from extension and if required from contents of the file to spell implicitly. The last key of the required keys is the `des[c[ription]]' key. It gives a short description of the filter mode which will displayed when type `aspell help'. The rest of the file consists of the keys `filter' and `option' to load filters are set various options. 12.1.1 Version Line ------------------- Each version line must start with `aspell' and be followed by a version, optionally prefixed by a relational operator. The relation operator can be one of `<', `<=', `=', `>=' or '>' for allowing Aspell version with version number being lower, lower or equal, equal to, greater or equal or greater than required version number, respectfully. If the relation operator is omitted `=' is assumed. 12.1.2 Magic Line ----------------- The magic line contains a description which requirements files have to fulfill in order to implicitly activate the entire mode at least one such line is required. Each magic line has the following format: MAGIC /<magic key>/<fileextention>[/<fileextention>] The magic key consist of three `:' separated fields. The first two are byte counts the last is a regular expression. The first byte count indicates the first byte the regular expression will be applied to the second byte count indicates the number of bytes to test against the regular expression. If mode selection should only occurred on basis of the listed file extensions the magic key should consist of the "<noregex>" special string. At least one <fileextention> is required per MAGIC line. <fileextention> may not be empty and should not contain a leading `.' as this is assumed implicitly. Multiple MAGIC lines are allowed. Modes may be extended limited by additional <label>.amf files located in -filter-path Thus file extensions may be prefixed by `+' or `-' to indicate that the entire extension has to be added ore removed from this <magic key> if neither is specified than a `+' is assumed implicitly. 12.1.3 Description Line ----------------------- The required description line will be printed when typing `aspell help'. Keep it as short as possible. Possible abbreviations are `des' and `desc'. 12.1.4 Filter and Option Lines ------------------------------ The `filter' and `option' keys load filters and set filter options. The value of the `filter' key is equal to the value of Aspell's `[add|rem]-filter' option. Each `option' line has the following format: OPTION <option> [<value>] The format of the <option> and <value> is the same format as found in the Aspell configuration file. File: aspell-dev.info, Node: Data Structures, Next: Mk-Src Script, Prev: Filter Modes, Up: Top 13 Data Structures ****************** Whenever possible you should try to use one of the data structures available. If the data structures do not provide enough functionality for your needs you should consider enhancing them rather than writing something from scratch. 13.1 Vector =========== The `vector' class is defined in `vector.hpp' and works the same way as the standard STL `vector' does except that it doesn't have as many constructors. 13.2 BasicList ============== `BasicList' is a simple list structure which can either be implemented as a singly or doubly linked list. It is defined in `basic_list.hpp'. 13.3 StringMap ============== `StringMap' is a associative array for strings. You should try to use this when ever possible to avoid code bloat. It is defined in `string_map.hpp'. 13.4 Hash Tables ================ Several hash tables are provided when `StringMap' is not appropriate. These hash tables provide a `hash_set', `hash_multiset', `hash_map' and `hash_multimap' which are very similar to SGI's STL implementation with a few exceptions. It is defined in `hash.hpp'. 13.5 BlockSList =============== `BlockSList' provided a pool of nodes which can be used for singly linked lists. It is defined in `block_slist.hpp'. File: aspell-dev.info, Node: Mk-Src Script, Next: How It All Works, Prev: Data Structures, Up: Top 14 Mk-Src Script **************** A good deal of interface code is automatically generated by the `mk-src.pl' Perl script. I am doing it this way to avoid having to write a lot of relative code for the C++ interface. This should also make adding interface for other languages a lot less tedious and will allow the interface to automatically take advantage of new Aspell functionality as it is made available. The `mk-src.pl' script uses `mk-src.in' as its input. 14.1 mk-src.in ============== NOTE: This section may not always be up to date since it is manually converted from the pod source. The format of `mk-src.in' is as follows: The following characters are literals: { } / '\ ' \n = > <items> <items> := (<item>\n)+ <items> := <category>:\ <name> {\n<details>\n} | <<tab>><details> <details> := <options>\n /\n <items> <options> := (<option>\n)* <option> := <key> [=> <value>] <<tab>> means everything should be indented by one tab See MkSrc::Info for a description of the categories and options 14.2 MkSrc::Info ================ `%info' The info array contains information on how to process the info in `mk-src.pl'. It has the following layout <catagory> => options => [] groups => [] # if undef than anything is accepted creates_type => "" # the object will create a new type # as specified proc => <impl type> => sub {} where <impl type> is one of: cc: for "aspell.h" header file cxx: for C++ interface implemented on top of cc interface native: for creation of header files used internally by aspell impl: for defination of functions declared in cc interface. the definations use the native hedaer files native_impl: for implementations of stuff declared in the native header files each proc sub should take the following argv $data: a subtree of $master_data $accum: <options> is one of: desc: description of the object prefix: posib err: the method may return an error condition c func: const: the method is a const member c only: only include in the external interface c impl headers: extra headers that need to be included in the C impl c impl: use this as the c impl instead of the default cxx impl: use this as the cxx impl instead of the default returns alt type: the constructor returns some type other than the object from which it is a member of no native: do not attemt to create a native implementation treat as object: treat as a object rather than a pointer The `%info' structure is initialized as follows: our %info = ( root => { options => [], groups => ['methods', 'group']}, methods => { # methods is a collection of methods which will be inserted into # a class after some simple substation rules. A $ will be # replaced with name of the class. options => ['strip', 'prefix', 'c impl headers'], groups => undef}, group => { # a group is a colection of objects which should be grouped together # this generally means they will be in the same source file options => ['no native'], groups => ['enum', 'struct', 'union', 'func', 'class', 'errors']}, enum => { # basic C enum options => ['desc', 'prefix'], creates_type => 'enum'}, struct => { # basic c struct options => ['desc', 'treat as object'], groups => undef, creates_type => 'struct',}, union => { # basic C union options => ['desc', 'treat as object'], groups => undef, creates_type => 'union'}, class => { # C++ class options => ['c impl headers'], groups => undef, creates_type => 'class'}, errors => {}, # possible errors method => { # A class method options => ['desc', 'posib err', 'c func', 'const', 'c only', 'c impl', 'cxx impl'], groups => undef}, constructor => { # A class constructor options => ['returns alt type', 'c impl', 'desc'], groups => 'types'}, destructor => { # A class destructor options => [], groups => undef}, ); In addition to the categories listed above a "methods" category by be specified in under the class category. A "methods" category is created for each methods group under the name "<methods name> methods". When groups is undefined a type name may be specified in place of a category. `%types' types contains a master list of all types. This includes basic types and ones created in `mk-src.in'. The basic types include: 'void', 'bool', 'pointer', 'double', 'string', 'encoded string', 'string obj', 'char', 'unsigned char', 'short', 'unsigned short', 'int', 'unsigned int', 'long', 'unsigned long' %methods `%methods' is used for holding the "methods" information 14.3 MkSrc::Util ================ This module contains various useful utility functions: `false' Returns 0. `true' Returns 1. `cmap EXPR LIST' Apply EXPR to each item in LIST and than concatenate the result into a string `one_of STR LIST' Returns true if LIST contains at least one of STR. `to_upper STR' Convert STR to all uppercase and substitute spaces with underscores. `to_lower STR' Convert STR to all lowercase and substitute spaces with underscores. `to_mixed STR' Convert STR to mixed case where each new word startes with a uppercase letter. For example "feed me" would become "FeedMe". 14.4 MkSrc::Read ================ `read' Read in `mk-src.in' and return a data structure which has the following format: <tree> <tree> := <options> data => <tree> where each tree represents an entry in mk-src.in. The following two options are always provided: name: the name of the entry type: the catagory or type name Additional options are the same as specified in %info 14.5 MKSrc::Create ================== `create_cc_file PARMS' Create a source file. Required Parms: type, dir, name, data Boolean Parms: header, cxx Optional Parms: namespace (required if cxx), pre_ext, accum `create_file FILENAME DATA' Writes DATA to FILENAME but only if DATA differs from the content of the file and the string: Automatically generated file. is present in the existing file if it already exists. 14.6 Code Generation Modes ========================== The code generation modes are currently one of the following: cc: Mode used to create types suitable for C interface cc_cxx: Like cc but typenames don't have a leading Aspell prefix cxx: Mode used to create types suitable for CXX interface native: Mode in which types are suitable for the internal implementation native_no_err: Like Native but with out PosibErr return types 14.7 MkSrc::CcHelper ==================== Helper functions used by interface generation code: to_c_return_type ITEM . c_error_cond ITEM . `make_func NAME @TYPES PARMS ; %ACCUM' Creates a function prototype Parms can be any of: mode: code generation mode `call_func NAME @TYPES PARMS ; %ACCUM' Return a string to call a func. Will prefix the function with return if the functions returns a non-void type; Parms can be any of: mode: code generation mode `to_type_name ITEM PARMS ; %ACCUM' Converts item into a type name. Parms can be any of: mode: code generation mode use_type: include the actual type use_name: include the name on the type pos: either "return" or "other" `make_desc DESC ; LEVEL' Make a C comment out of DESC optionally indenting it LEVEL spaces. `make_c_method CLASS ITEM PARMS ; %ACCUM' Create the phototype for a C method which is really a function. Parms is any of: mode: code generation mode no_aspell: if true do not include aspell in the name this_name: name for the parameter representing the current object `call_c_method CLASS ITEM PARMS ; %ACCUM' Like make_c_method but instead returns the appropriate string to call the function. If the function returns a non-void type the string will be prefixed with a return statement. `form_c_method CLASS ITEM PARMS ; %ACCUM' Like make_c_method except that it returns the array: ($func, $data, $parms, $accum) which is suitable for passing into make_func. It will return an empty array if it can not make a method from ITEM. `make_cxx_method ITEM PARMS ; %ACCUM' Create the phototype for a C++ method. Parms is one of: mode: code generation mode File: aspell-dev.info, Node: How It All Works, Next: Part 1 - Compiled Dictionary Format, Prev: Mk-Src Script, Up: Top 15 How It All Works ******************* The details of how Aspell really works is a mystery to most users who want to participate in developing and improving Aspell, so it is best to fully explain Aspell's core algorithms and data structures to you. In explaining these, it is hoped to bring prospective developers up to speed more quickly and also help you understand the amount of thought put into this. In time, you may be able to improve Aspell even more. There are many details to explain here, so these will need explaining in small segments. * Menu: * Part 1 - Compiled Dictionary Format:: * Part 2 - Quickly Finding Similar Soundslike:: * Part 3:: File: aspell-dev.info, Node: Part 1 - Compiled Dictionary Format, Next: Part 2 - Quickly Finding Similar Soundslike, Prev: How It All Works, Up: How It All Works 15.1 Part 1 - The Compiled Dictionary Format ============================================ In this part you will see how the data is laid out in the compiled dictionary for Aspell 0.60. See source file `readonly_ws.cpp'. Aspell's main compiled wordlist dictionary file is made as follows: * header * jump table for editdist 1 * jump table for editdist 2 * data block * hash table There is nothing particularly interesting about the header. Just a bunch of meta information. The jump tables are described in the next section ... Words in the data block are grouped based on the soundslike. Each group is as follows: <8 bit: offset to next item><8 bit: soundslike size><soundslike> <null><words> Each word group is as follows: <8 bit: flags><8 bit: offset to next word><8 bit: word size><word><null> [<affix info><null>] The offset for the final word in each group points to the next word in the following group. The offset for the final word and soundslike group in the dictionary is 0. There is some provisions for additional info to be stored with the word but for simplicity, it's left out here. If soundslike data is not used then the soundslike block it not used. This format makes it easy to iterate over the data without using the hash table. Each soundslike group can be a maximum of about 256 bytes. If this limit is reached then the soundslike group is split. Using 2 bytes for the soundslike offset would of solved this problem however 256 bytes is normally sufficient, thus I would of wasted some space by using an extra byte. More importantly, Using 2 bytes means I would of had to worry about alignment issues. The soundslike groups are sorted in more or less alphabetic order. The hash table is a simple open address hash table. The key is the dictionary word in all lowercase form with all accents removed (what is known as the "clean" form of the word). The value stored in the table is a 32-bit offset to the beginning of the word. A 32-bit integer offset is used rather than a pointer so that the compiled dictionary can be mmaped to make loading the dictionary very fast and so that the memory can be shared between processed, and on 64 bit platforms using pointers would have doubled the size of the hash table. Additional information for each word can be derived from each offset: word size: offset - 1 offset to next word: offset - 2 flags: offset - 3 I use helper functions for getting this information. Doing it this way rather than having a data structure is slightly evil, but admittedly, I have my reasons. In the next section, you will see how Aspell uses the jump tables to search the list for _soundslike_ with _edit-distance_ of 1 or 2. File: aspell-dev.info, Node: Part 2 - Quickly Finding Similar Soundslike, Next: Part 3, Prev: Part 1 - Compiled Dictionary Format, Up: How It All Works 15.2 Part 2 - Quickly Finding Similar Soundslike ================================================ In order for Aspell to find suggestions for a misspelled word Aspell 1) creates a list of candidate words, 2) scores them, and 3) returns the most likely candidates. One of the ways Aspell finds candidate words is to look for all words with a soundslike which is of a small edit distance from the soundslike of the original word. The edit distance is the total number of deletions, insertions, exchanges, or adjacent swaps needed to make one string equivalent to the other. The exact distance chosen is either 1 or 2 depending on a number of factors. In this part I will focus on how Aspell find all such soundslike efficiently and how the jump tables play a key role. This section will focus on how Aspell finds all such soundslike efficiently and how the jump tables play a key role. The naive way to scan the list for all possible soundslike is to compute the edit-distance of every soundslike in the dictionary and then keep the ones within the threshold. This is exactly what Aspell did prior to 0.60. before a faster method was created. When a fast enough edit distance function is used this method turns out not to be unbearably slow, at least for English, but for other languages, with large word lists and no soundslike, this can be slow due to the number of items that need to be scanned. Aspell uses a special edit distance function which gives up if the distance is larger than the threshold, thus making it very fast. The basic algorithm is as follows: limit_edit_distance(A,B,limit) = ed(A,B,0) where ed(A,B,d) = d if A & B is empty. = infinity if d > limit = ed(A[2..],B[2..], d) if A[1] == B[1] = min ( ed(A[2..],B[2..], d+1), ed(A, B[2..], d+1), ed(A[2..],B, d+1) ) otherwise However the algorithm used also allows for swaps and is not recursive. Specialized versions are provided for an edit distance of one and two. The running time is asymptotically bounded above by `(3^l)*n' where `l' is the limit and `n' is the maximum of `strlen(A),strlen(B)'. Based on informal tests, the `n' does not really matter and the running time is more like `(3^l)'. For complete details on this algorithm see the files `leditdist.hpp' and `leditdist.cpp' in the source distribution under `modules/speller/default'. So, by exploiting the properties of `limit_edit_distance' it is possible to avoid having to look at many of the soundslikes in the dictionary. `Limit_edit_distance' is efficient because in many cases, it does not have to look at the entire word before it can determine that it isn't within the given threshold, and then by having it return the last position looked at, _p_, it is possible to avoid having to look at similar soundslike which are not within the threshold. That is, if two soundslike are the same up to the position `p', then neither of them are within the given threshold. Aspell 0.60 exploits this property by using jump tables. Each entry in the jump table contains two fields: the first `N' letters of a soundslike, and an offset. The entries are sorted in lexicographic order based on the raw byte value. Aspell maintains two jump tables. The first table contains the first 2 letters of a soundslike and the offset points into the second jump table. The second table contains the first 3 letters of a soundslike where the offset points to the location of the soundslike in the data block. The soundslike in the datablock are sorted so that a linear scan can be used to find all soundslike with the same prefix. If the `limit_edit_distance' stops before reaching the end of a _"soundslike"_ in one of the jump tables then it is possible to skip all the soundslike in the data block with the same prefix. Thus, the scan for all _soundslike_ within a given edit distance goes something like this: 1. Compare the entry in the first jump table using `limit_edit_distance'. If the `limit_edit_distance' scanned passed the end of the word, then go to the first entry in the second jump table with the same prefix, otherwise go to the next entry in the first jump table and repeat. 2. Compare the entry in the second jump table. If the `limit_edit_distance' passed the end of the word, then go to the first _soundslike_ in the data block with this prefix, otherwise if the first two letters of the next entry are the same as the current one go to it and repeat. If the first two letters are not the same then go to the next entry in the first jump table and repeat step 1. 3. Compare the _soundslike_ in the data block. If the edit distance is within the target distance, then add the word to the candidate list, otherwise don't. Let `N' be the position where `limit_edit_distance' stopped, (starting at 0). If `N' is less than 6, then skip over any soundslike that have the same first `N + 1' letters. If after skipping over any similar _soundslike_ the next _soundslike_ does not have the same first three letters, then go to the next entry in the second jump table and repeat step 2, otherwise repeat this step with the next _soundslike_. The part of skipping over _soundslike_ with the first `N + 1' letters in step 3 were added in Aspell 0.60.3. The function responsible for most of this is found in function `ReadOnlyDict::SoundslikeElements::next' which is found in file `readonly_ws.cpp'. The next part will describe how Aspell deals with _soundslike_ lookup when affix compression is involved. File: aspell-dev.info, Node: Part 3, Next: Copying, Prev: Part 2 - Quickly Finding Similar Soundslike, Up: How It All Works 15.3 Part 3 =========== Not written yet. File: aspell-dev.info, Node: Copying, Prev: Part 3, Up: Top Appendix A Copying ****************** Copyright (C) 2002, 2003, 2004, 2006 Kevin Atkinson. Permission is granted to copy, distribute and/or modify this document under the terms of the GNU Free Documentation License, Version 1.1 or any later version published by the Free Software Foundation; with no Invariant Sections, no Front-Cover Texts and no Back-Cover Texts. A copy of the license is included in the section entitled "GNU Free Documentation License". * Menu: * GNU Free Documentation License:: File: aspell-dev.info, Node: GNU Free Documentation License, Up: Copying A.1 GNU Free Documentation License ================================== Version 1.2, November 2002 Copyright (C) 2000,2001,2002 Free Software Foundation, Inc. 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA Everyone is permitted to copy and distribute verbatim copies of this license document, but changing it is not allowed. 0. PREAMBLE The purpose of this License is to make a manual, textbook, or other functional and useful document "free" in the sense of freedom: to assure everyone the effective freedom to copy and redistribute it, with or without modifying it, either commercially or noncommercially. Secondarily, this License preserves for the author and publisher a way to get credit for their work, while not being considered responsible for modifications made by others. This License is a kind of "copyleft", which means that derivative works of the document must themselves be free in the same sense. It complements the GNU General Public License, which is a copyleft license designed for free software. We have designed this License in order to use it for manuals for free software, because free software needs free documentation: a free program should come with manuals providing the same freedoms that the software does. But this License is not limited to software manuals; it can be used for any textual work, regardless of subject matter or whether it is published as a printed book. We recommend this License principally for works whose purpose is instruction or reference. 1. APPLICABILITY AND DEFINITIONS This License applies to any manual or other work, in any medium, that contains a notice placed by the copyright holder saying it can be distributed under the terms of this License. Such a notice grants a world-wide, royalty-free license, unlimited in duration, to use that work under the conditions stated herein. The "Document", below, refers to any such manual or work. Any member of the public is a licensee, and is addressed as "you". You accept the license if you copy, modify or distribute the work in a way requiring permission under copyright law. A "Modified Version" of the Document means any work containing the Document or a portion of it, either copied verbatim, or with modifications and/or translated into another language. A "Secondary Section" is a named appendix or a front-matter section of the Document that deals exclusively with the relationship of the publishers or authors of the Document to the Document's overall subject (or to related matters) and contains nothing that could fall directly within that overall subject. (Thus, if the Document is in part a textbook of mathematics, a Secondary Section may not explain any mathematics.) The relationship could be a matter of historical connection with the subject or with related matters, or of legal, commercial, philosophical, ethical or political position regarding them. The "Invariant Sections" are certain Secondary Sections whose titles are designated, as being those of Invariant Sections, in the notice that says that the Document is released under this License. If a section does not fit the above definition of Secondary then it is not allowed to be designated as Invariant. The Document may contain zero Invariant Sections. If the Document does not identify any Invariant Sections then there are none. The "Cover Texts" are certain short passages of text that are listed, as Front-Cover Texts or Back-Cover Texts, in the notice that says that the Document is released under this License. A Front-Cover Text may be at most 5 words, and a Back-Cover Text may be at most 25 words. A "Transparent" copy of the Document means a machine-readable copy, represented in a format whose specification is available to the general public, that is suitable for revising the document straightforwardly with generic text editors or (for images composed of pixels) generic paint programs or (for drawings) some widely available drawing editor, and that is suitable for input to text formatters or for automatic translation to a variety of formats suitable for input to text formatters. A copy made in an otherwise Transparent file format whose markup, or absence of markup, has been arranged to thwart or discourage subsequent modification by readers is not Transparent. An image format is not Transparent if used for any substantial amount of text. A copy that is not "Transparent" is called "Opaque". Examples of suitable formats for Transparent copies include plain ASCII without markup, Texinfo input format, LaTeX input format, SGML or XML using a publicly available DTD, and standard-conforming simple HTML, PostScript or PDF designed for human modification. Examples of transparent image formats include PNG, XCF and JPG. Opaque formats include proprietary formats that can be read and edited only by proprietary word processors, SGML or XML for which the DTD and/or processing tools are not generally available, and the machine-generated HTML, PostScript or PDF produced by some word processors for output purposes only. The "Title Page" means, for a printed book, the title page itself, plus such following pages as are needed to hold, legibly, the material this License requires to appear in the title page. For works in formats which do not have any title page as such, "Title Page" means the text near the most prominent appearance of the work's title, preceding the beginning of the body of the text. A section "Entitled XYZ" means a named subunit of the Document whose title either is precisely XYZ or contains XYZ in parentheses following text that translates XYZ in another language. (Here XYZ stands for a specific section name mentioned below, such as "Acknowledgements", "Dedications", "Endorsements", or "History".) To "Preserve the Title" of such a section when you modify the Document means that it remains a section "Entitled XYZ" according to this definition. The Document may include Warranty Disclaimers next to the notice which states that this License applies to the Document. These Warranty Disclaimers are considered to be included by reference in this License, but only as regards disclaiming warranties: any other implication that these Warranty Disclaimers may have is void and has no effect on the meaning of this License. 2. VERBATIM COPYING You may copy and distribute the Document in any medium, either commercially or noncommercially, provided that this License, the copyright notices, and the license notice saying this License applies to the Document are reproduced in all copies, and that you add no other conditions whatsoever to those of this License. You may not use technical measures to obstruct or control the reading or further copying of the copies you make or distribute. However, you may accept compensation in exchange for copies. If you distribute a large enough number of copies you must also follow the conditions in section 3. You may also lend copies, under the same conditions stated above, and you may publicly display copies. 3. COPYING IN QUANTITY If you publish printed copies (or copies in media that commonly have printed covers) of the Document, numbering more than 100, and the Document's license notice requires Cover Texts, you must enclose the copies in covers that carry, clearly and legibly, all these Cover Texts: Front-Cover Texts on the front cover, and Back-Cover Texts on the back cover. Both covers must also clearly and legibly identify you as the publisher of these copies. The front cover must present the full title with all words of the title equally prominent and visible. You may add other material on the covers in addition. Copying with changes limited to the covers, as long as they preserve the title of the Document and satisfy these conditions, can be treated as verbatim copying in other respects. If the required texts for either cover are too voluminous to fit legibly, you should put the first ones listed (as many as fit reasonably) on the actual cover, and continue the rest onto adjacent pages. If you publish or distribute Opaque copies of the Document numbering more than 100, you must either include a machine-readable Transparent copy along with each Opaque copy, or state in or with each Opaque copy a computer-network location from which the general network-using public has access to download using public-standard network protocols a complete Transparent copy of the Document, free of added material. If you use the latter option, you must take reasonably prudent steps, when you begin distribution of Opaque copies in quantity, to ensure that this Transparent copy will remain thus accessible at the stated location until at least one year after the last time you distribute an Opaque copy (directly or through your agents or retailers) of that edition to the public. It is requested, but not required, that you contact the authors of the Document well before redistributing any large number of copies, to give them a chance to provide you with an updated version of the Document. 4. MODIFICATIONS You may copy and distribute a Modified Version of the Document under the conditions of sections 2 and 3 above, provided that you release the Modified Version under precisely this License, with the Modified Version filling the role of the Document, thus licensing distribution and modification of the Modified Version to whoever possesses a copy of it. In addition, you must do these things in the Modified Version: A. Use in the Title Page (and on the covers, if any) a title distinct from that of the Document, and from those of previous versions (which should, if there were any, be listed in the History section of the Document). You may use the same title as a previous version if the original publisher of that version gives permission. B. List on the Title Page, as authors, one or more persons or entities responsible for authorship of the modifications in the Modified Version, together with at least five of the principal authors of the Document (all of its principal authors, if it has fewer than five), unless they release you from this requirement. C. State on the Title page the name of the publisher of the Modified Version, as the publisher. D. Preserve all the copyright notices of the Document. E. Add an appropriate copyright notice for your modifications adjacent to the other copyright notices. F. Include, immediately after the copyright notices, a license notice giving the public permission to use the Modified Version under the terms of this License, in the form shown in the Addendum below. G. Preserve in that license notice the full lists of Invariant Sections and required Cover Texts given in the Document's license notice. H. Include an unaltered copy of this License. I. Preserve the section Entitled "History", Preserve its Title, and add to it an item stating at least the title, year, new authors, and publisher of the Modified Version as given on the Title Page. If there is no section Entitled "History" in the Document, create one stating the title, year, authors, and publisher of the Document as given on its Title Page, then add an item describing the Modified Version as stated in the previous sentence. J. Preserve the network location, if any, given in the Document for public access to a Transparent copy of the Document, and likewise the network locations given in the Document for previous versions it was based on. These may be placed in the "History" section. You may omit a network location for a work that was published at least four years before the Document itself, or if the original publisher of the version it refers to gives permission. K. For any section Entitled "Acknowledgements" or "Dedications", Preserve the Title of the section, and preserve in the section all the substance and tone of each of the contributor acknowledgements and/or dedications given therein. L. Preserve all the Invariant Sections of the Document, unaltered in their text and in their titles. Section numbers or the equivalent are not considered part of the section titles. M. Delete any section Entitled "Endorsements". Such a section may not be included in the Modified Version. N. Do not retitle any existing section to be Entitled "Endorsements" or to conflict in title with any Invariant Section. O. Preserve any Warranty Disclaimers. If the Modified Version includes new front-matter sections or appendices that qualify as Secondary Sections and contain no material copied from the Document, you may at your option designate some or all of these sections as invariant. To do this, add their titles to the list of Invariant Sections in the Modified Version's license notice. These titles must be distinct from any other section titles. You may add a section Entitled "Endorsements", provided it contains nothing but endorsements of your Modified Version by various parties--for example, statements of peer review or that the text has been approved by an organization as the authoritative definition of a standard. You may add a passage of up to five words as a Front-Cover Text, and a passage of up to 25 words as a Back-Cover Text, to the end of the list of Cover Texts in the Modified Version. Only one passage of Front-Cover Text and one of Back-Cover Text may be added by (or through arrangements made by) any one entity. If the Document already includes a cover text for the same cover, previously added by you or by arrangement made by the same entity you are acting on behalf of, you may not add another; but you may replace the old one, on explicit permission from the previous publisher that added the old one. The author(s) and publisher(s) of the Document do not by this License give permission to use their names for publicity for or to assert or imply endorsement of any Modified Version. 5. COMBINING DOCUMENTS You may combine the Document with other documents released under this License, under the terms defined in section 4 above for modified versions, provided that you include in the combination all of the Invariant Sections of all of the original documents, unmodified, and list them all as Invariant Sections of your combined work in its license notice, and that you preserve all their Warranty Disclaimers. The combined work need only contain one copy of this License, and multiple identical Invariant Sections may be replaced with a single copy. If there are multiple Invariant Sections with the same name but different contents, make the title of each such section unique by adding at the end of it, in parentheses, the name of the original author or publisher of that section if known, or else a unique number. Make the same adjustment to the section titles in the list of Invariant Sections in the license notice of the combined work. In the combination, you must combine any sections Entitled "History" in the various original documents, forming one section Entitled "History"; likewise combine any sections Entitled "Acknowledgements", and any sections Entitled "Dedications". You must delete all sections Entitled "Endorsements." 6. COLLECTIONS OF DOCUMENTS You may make a collection consisting of the Document and other documents released under this License, and replace the individual copies of this License in the various documents with a single copy that is included in the collection, provided that you follow the rules of this License for verbatim copying of each of the documents in all other respects. You may extract a single document from such a collection, and distribute it individually under this License, provided you insert a copy of this License into the extracted document, and follow this License in all other respects regarding verbatim copying of that document. 7. AGGREGATION WITH INDEPENDENT WORKS A compilation of the Document or its derivatives with other separate and independent documents or works, in or on a volume of a storage or distribution medium, is called an "aggregate" if the copyright resulting from the compilation is not used to limit the legal rights of the compilation's users beyond what the individual works permit. When the Document is included in an aggregate, this License does not apply to the other works in the aggregate which are not themselves derivative works of the Document. If the Cover Text requirement of section 3 is applicable to these copies of the Document, then if the Document is less than one half of the entire aggregate, the Document's Cover Texts may be placed on covers that bracket the Document within the aggregate, or the electronic equivalent of covers if the Document is in electronic form. Otherwise they must appear on printed covers that bracket the whole aggregate. 8. TRANSLATION Translation is considered a kind of modification, so you may distribute translations of the Document under the terms of section 4. Replacing Invariant Sections with translations requires special permission from their copyright holders, but you may include translations of some or all Invariant Sections in addition to the original versions of these Invariant Sections. You may include a translation of this License, and all the license notices in the Document, and any Warranty Disclaimers, provided that you also include the original English version of this License and the original versions of those notices and disclaimers. In case of a disagreement between the translation and the original version of this License or a notice or disclaimer, the original version will prevail. If a section in the Document is Entitled "Acknowledgements", "Dedications", or "History", the requirement (section 4) to Preserve its Title (section 1) will typically require changing the actual title. 9. TERMINATION You may not copy, modify, sublicense, or distribute the Document except as expressly provided for under this License. Any other attempt to copy, modify, sublicense or distribute the Document is void, and will automatically terminate your rights under this License. However, parties who have received copies, or rights, from you under this License will not have their licenses terminated so long as such parties remain in full compliance. 10. FUTURE REVISIONS OF THIS LICENSE The Free Software Foundation may publish new, revised versions of the GNU Free Documentation License from time to time. Such new versions will be similar in spirit to the present version, but may differ in detail to address new problems or concerns. See `http://www.gnu.org/copyleft/'. Each version of the License is given a distinguishing version number. If the Document specifies that a particular numbered version of this License "or any later version" applies to it, you have the option of following the terms and conditions either of that specified version or of any later version that has been published (not as a draft) by the Free Software Foundation. If the Document does not specify a version number of this License, you may choose any version ever published (not as a draft) by the Free Software Foundation. A.1.1 ADDENDUM: How to use this License for your documents ---------------------------------------------------------- To use this License in a document you have written, include a copy of the License in the document and put the following copyright and license notices just after the title page: Copyright (C) YEAR YOUR NAME. Permission is granted to copy, distribute and/or modify this document under the terms of the GNU Free Documentation License, Version 1.2 or any later version published by the Free Software Foundation; with no Invariant Sections, no Front-Cover Texts, and no Back-Cover Texts. A copy of the license is included in the section entitled ``GNU Free Documentation License''. If you have Invariant Sections, Front-Cover Texts and Back-Cover Texts, replace the "with...Texts." line with this: with the Invariant Sections being LIST THEIR TITLES, with the Front-Cover Texts being LIST, and with the Back-Cover Texts being LIST. If you have Invariant Sections without Cover Texts, or some other combination of the three, merge those two alternatives to suit the situation. If your document contains nontrivial examples of program code, we recommend releasing these examples in parallel under your choice of free software license, such as the GNU General Public License, to permit their use in free software.
Generated by $Id: phpMan.php,v 4.55 2007/09/05 04:42:51 chedong Exp $ Author: Che Dong
On Apache/2.4.6 (CentOS)
Under GNU General Public License
2025-01-21 10:37 @127.0.0.1 CrawledBy Mozilla/5.0 AppleWebKit/537.36 (KHTML, like Gecko; compatible; ClaudeBot/1.0; +claudebot@anthropic.com)