FreshLIb user guide and howto
1. Work with dynamic memory.
1.1. Use with heap manager
FreshLib provides uniform and OS independent way to work with dynamic memory blocks. The API is simple - there are only 3 procedures:
proc GetMem, .size
- allocates new block with size at least [.size]
and returns a pointer to this
block in EAX. In case of error the procedure returns EAX=0 and CF=1.
proc FreeMem, .ptr
- frees already allocated by GetMem block of memory.
proc ResizeMem, .ptr, .newsize
- changes the size of some memory block, allocated by GetMem.
The implementation of these procedures differs on the different OSes. In Win32, these procedures are implemented using Win32 heap functions. In Linux, using libc functions.
There is an another approach to the memory management - FreshLib has its own implementation of a heap manager,
that is OS independent and can be used in multiply operating systems.
What heap management code will be used is defined through the HeapManager
preprocessor symbol to "ASM".
If HeapManager
is set to OS
, the OS specific memory management will be used.
If HeapManager
is set to ASM
the FreshLib heap manager will be compiled.
The FreshLib heap manager must always be used for OSes that has no their own method of heap management. For example KolibriOS port of the library uses this heap manager by default.
The heap manager procedures in FreshLib are thread safe.
1.2. Use without heap manager
The heap management is not mandatory and can be disabled at all. In this case the application will be able to allocate the whole needed memory as one big chunk.
There is only one procedure about this approach:
proc SpaceAllocate, .size
- allocates memory with the specified size. In this case, the start
address of the allocated memory is placed in the global variable _MemoryFreeSpace
and will not be
changed during the program execution. Only the size of the available block of RAM can be changed by
the above procedure.
Unfortunately, big amount of FreshLib functionality depends on the heap manager. This way, if you are using this memory management approach, this functionality can not be used in the user program.
That is why, generally, this approach is not recommended for wide use. Nevertheless, there are some situations (for example relatively simple program, that needs big, contiguous block of memory) where such simple approach can give you big performance gain without the mentioned disadvantages.
2. Work with dynamic strings
The strings in FreshLib are dynamic - their length is changed in order to hold the whole information user put into the string.
Because of this, the strings are represented not by their address, but by handle. However, most of the procedures in the library can operate with memory pointers as well.
The strings in FreshLib are always UTF-8 encoded.
2.1. Internal string format
The library distinguish the handles from the pointers, because of the two most significant bits (30th and 31st) of the handle are always set to 1. (this approach can rarely create some problem with systems where such high addresses are used, but usually it is OK).
In order to create new string, the user should use the procedures StrNew
or StrDup
. The first
creates new empty string. The second, creates new string with content copied from some existing
string.
Some of the strings processing procedures also can create new string and return it as a result value.
The format of the strings in memory is following:
Offset | Name | Size | Description |
‒8 | .Capacity | dword | Contains the amount of memory buffer allocated for the string. |
‒4 | .Length | dword | Contains the actual length of the string in bytes, not counting the terminating zeros. All StrLib procedures will update this value properly when changed. |
0 | .String | N | Contains the string data in UTF-8 encoding. |
N | ... | dword | At least one dword of 0 terminating the string. |
2.2. Short list of the string manipulating procedures
Here is a table with most used string functions with short description:
Procedure | Arguments | Operation | Notes |
StrNew
|
..... | Creates new string | ..... |
StrDup
|
.hString | Creates new string from existing one. | ..... |
StrPtr
|
.hString | Returns a pointer to the current string memory. | ..... |
StrDel
|
.hString | Deletes some string. | Preserves all registers and flags. |
StrLen
|
.hString | Returns the length of the string in bytes. This procedure is very fast for direct pointers and instant for FreshLib strings. | ..... |
StrCopy
|
.dest, .source | Copies the content of the [.source] string to the [.dest] string.
|
..... |
StrCat
|
.dest, .source | Concatenate the content of [.source] at the end of [.dest] string.
|
..... |
StrCopyPart
|
.dest, .source, .pos, .len | Copies a part of [.source] string - from byte [.pos] with length [.len] to the [.dest] string.
|
..... |
StrExtract
|
.string, .pos, .len | Extracts a part of the string [.string] (from byte [.pos] and with length [.len] ) and returns it as
a new created string.
|
..... |
StrSplit
|
.hString, .pos | Splits the string by two parts. The left part remains in [.hString] and the right part remains in new
created string, returned by the procedure.
|
..... |
StrTrim
|
.hString, .pos | Trims the string at position [.pos] The remaining of the string is lost.
|
..... |
StrInsert
|
.dest, .source, .pos | Inserts the string [.source] in the string [.dest] at byte [.pos] .
|
..... |
NumToStr
|
.num, .flags | Converts the 32bit number [.num] into a new string, depending on the flags [.flags] .
|
Flags: ((ntsSigned | ntsUnsigned)+(ntsFixedWidth)) + (ntsBin | ntsQuad | ntsOct | ntsDec | ntsHex) |
StrToNumEx
|
.hString | Converts a string containing FASM formated number into a number. | ..... |
StrCharCat
|
.hString, .char | Concatenates to the string [.hString] up to 4 bytes of ascii (UTF-8) characters from [.char] .
|
..... |
StrCharInsert
|
.hString, .char, .pos | Inserts up to 4 characters from [.char] into the string [.hString] at the position [.pos] .
|
..... |
StrClipSpacesR
|
.hString | Removes all trailing spaces from the string [.hString] .
|
..... |
StrClipSpacesL
|
.hString | Removes all leading spaces from the string [.hString] .
|
..... |
StrCleanDupSpaces
|
.hString | Removes all duplicated spaces from [.hString] and leaves only one space on this place.
|
..... |
StrHash
|
.hString | Computes and returns FNV-1b hash of the string [.hString] .
|
..... |
DataHash
|
.ptrData, .len | Computes the FNV-1b hash of the memory area from [.ptrData] with length [.len] .
|
..... |
StrURLEncode
|
.hString | Encodes the string in [.hString] in URL encoding.
|
..... |
StrURLDecode
|
.hString | Decodes URL encoded string in place. | ..... |
StrLenUtf8
|
.hString | Computes the length in chars of the UTF-8 encoded string in [.hString] This operation is relatively slow
because the string must be scanned byte by byte.
|
..... |
StrOffsUtf8
|
.hString, .pos | Returns the byte offset of the character position [.pos] in the UTF-8 encoded string [.hString] .
|
..... |
StrExpandTabs
|
.hString, .tabstop | Converts all tabs in the string [.hString] into a spaces using [.tabstop] as a tab distance.
|
..... |
StrIP2Num
|
.hString | Converts the string representation of IP address into a 32bit number. | ..... |
IP2Str
|
.ip | Converts numeric representation of an IP number [.ip] into a string.
|
..... |
StrEncodeHTML
|
.hString | Encodes the string [.hString] in HTML encoding.
|
..... |
StrDecodeHTML
|
.hString | Decodes the HTML encoded string [.hString] .
|
..... |
DateTimeToStr
|
.pDateTime, .format | Converts TDateTime structure into a string, depending on [.format] flags.
|
..... |
StrMD5
|
.hString | Computes MD5 sum of the string [.hString] .
|
..... |
StrSplitList
|
.hString, .Separator, .fAutoEmpty | Creates dynamic array of dword, filled with the parts of the string [.hString] , splitted on the character
[.Separator] . The returned array must be freed after use with: stdcall ListFree, [RetPtr], StrDel .
|
..... |
AnsiToUtf8
|
.SrcEncoding, .hString | Converts an ANSI, [.SrcEncoding] encoded string into UTF-8 encoding.
|
WIN1251, CP866, KOI8R and KOI8U are currently supported, but FreshLib has tool that creates encoding tables from UNICODE mapping tables from http://www.unicode.org/Public/MAPPINGS/ |
Utf8ToAnsi
|
.hString, .DstEncoding | Converts UTF-8 encoded string into an ANSI encoded with encoding [.DstEncoding]
|
..... |
2.3. String procedures example
Here is some simple example demonstrating the use of several string functions:
iglobal
MyConst1 db 'Mickey ', 0
MyConst2 db 'Mouse ', 0
endg
stdcall StrNew
mov ebx, eax
stdcall StrDup, MyConst1
stdcall StrCat, eax, MyConst2 ; note, that StrCat does not change any registers.
stdcall StrCat, ebx, eax
stdcall StrDel, eax ; StrDel also preserves the flags.
3. Using dynamic libraries other than the standard.
FreshLib creates all the import tables needed for the program automatically.
Usually, the functions imported from external DLLs are not supposed to be called from the user code, because such calls are not portable.
But sometimes, the programmer needs some functionality that is accessible only through an external library. There are two cases:
1. The internal library is available only for one particular OS. In this case, the result program will be not portable and nothing can be done.
2. There are versions of the same library for all target platforms. This is the best variant that will allow to keep the created program portable. But in this variant, there is a catch. It requires the calling conventions of the different OS versions of the library to be the same.
In order to use external DLLs, the one will need a file with definitions of the imported functions placed somewhere in the project directory. This file should contain a list with the functions exported from the DLL using the FreshLib macro 'import_proto'.
macro is defined for Linux and Windows, following way:
macro import_proto library, [iname, arg, ename]
Where:
library
is the name of the DLL;
iname
is the name of the function as it will appears in the FreshLib program.
arg
is an optional list with the function's arguments. They will be used for function call argument hinting
ename
is a string with the name of the function as exported from the DLL.
The only difference between the Windows and Linux version is the value of library field. In Windows it is the name of the DLL without the extension. In Linux it is a string with the full name of the shared library.
Here are two examples of this macro use for sqlite3 library for Windows and Linux (here are only the first 4 function definitions):
In Windows:
import_proto sqlite3, \
sqliteAggregateContext, <.context, .nbytes>, 'sqlite3_aggregate_context', \
sqliteAggregateCount_OLD, <VOID>, 'sqlite3_aggregate_count', \
sqliteAutoExtension, <.ptrCallback>, 'sqlite3_auto_extension', \
sqliteBackupFinish, <.ptrBackup>, 'sqlite3_backup_finish', \
In Linux:
import_proto 'libsqlite3.so.0', \
sqliteAggregateContext, <.context, .nbytes>, 'sqlite3_aggregate_context', \
sqliteAggregateCount_OLD, <VOID>, 'sqlite3_aggregate_count', \
sqliteAutoExtension, <.ptrCallback>, 'sqlite3_auto_extension', \
sqliteBackupFinish, <.ptrBackup>, 'sqlite3_backup_finish', \
In the argument definitions, there are two special values: <NONE>
means that the arguments are unknown and <VOID>
means the function does not has any arguments. All arguments must start with '.', i.e. to be defined as local labels.
In the above example, the function sqlite3_aggregate_context
has two arguments .context
and .nbytes
and is
imported with the name sqliteAggregateContext
.
If several definition files for different OSes are created, they have to be placed in separate directories, named after the OS they are targeted for. For the above example the first file will be placed in "myproject/Win32/sqlite.inc" and the second in "myproject/Linux/sqlite.inc".
If only one OS definition file is created, it may be placed anywhere.
The use of these definition files is with the macro uses
, placed somewhere in your code that uses this library. The
macro is defined following way:
macro uses [libname]
Where libname
is the name of the library in format: LIBRARY_NAME[:definition_filename]
. The filename of the
definition file is optional and if omitted is created automatically as '%lib%/imports/%TargetOS%/LIBRARY_NAME.inc
If several definition filenames are created for different OSes, the uses macro must contain %TargetOS% environment variable. For example:
uses sqlite3:'myproject/%TargetOS%/sqlite3.inc'
Defined this way functions can be used in the program from everywhere and FreshLib will automatically create the import table of the executable.
Notice, that only the functions that are used will be imported. This way, it is always safe to use uses
macro.
It will never create over-bloated import table.
Last modified on: 13.01.2016 19:03:08