2019-07-15 14:21:50 +02:00
[< back to index ](../doc_index.md )
2018-07-12 18:30:35 +02:00
# Preprocessor
The Millfork preprocessor does 2 things:
* filters lines in the input file according to current target's features
* injects the target's feature values as constants into the current file
Despite its similarity to the C preprocessor, it's much more restricted in its power:
* no file inclusion
* no macros
* separate namespaces for the preprocessor and the language (you need to use `#use` to use a preprocessor constant in the code)
2019-06-14 15:33:17 +02:00
Preprocessor directives by default start with `#` .
To avoid conflicts with C preprocessor (for users wishing to use it), it is also possible to replace `#` with `$$` .
2018-07-12 18:30:35 +02:00
### Defining feature values
Feature values are defined in the `[define]` section of the platform definition file.
Each value is a signed 64-bit integer number.
Example:
[define]
WIDESCREEN=1
You can also define feature values using the `-D` command line option.
### Built-in features
2018-08-08 13:44:30 +02:00
The following features are defined based on the chosen CPU and compilation options:
2018-12-29 20:16:05 +01:00
* `MILLFORK_VERSION` – defined since 0.3.4, contains the version of the compiler: for version x.y.z, the value is 10000x+100y+z
2018-07-30 18:16:50 +02:00
* `ARCH_6502` – 1 if compiling for 6502, 0 otherwise
2018-07-12 18:30:35 +02:00
2018-08-08 13:44:30 +02:00
* `ARCH_I80` – 1 if compiling for Intel 8080-like processor, 0 otherwise
2018-07-30 18:16:50 +02:00
2019-05-31 17:03:35 +02:00
* `ARCH_X86` – 1 if compiling for Intel 8086-like processor, 0 otherwise
2019-06-12 12:06:02 +02:00
* `CPU_65C02` , `CPU_65CE02` , `CPU_65816` , `CPU_HUC6280` , `CPU_8080` , `CPU_8085` , `CPU_GAMEBOY` , `CPU_Z80` , `CPU_8086`
2018-08-08 13:44:30 +02:00
– 1 if compiling for the exact given processor, 0 otherwise
2018-07-30 18:16:50 +02:00
2018-08-08 13:44:30 +02:00
* `CPU_6502` – 1 if compiling for any pre-65C02 6502-like processor, 0 otherwise
* `CPUFEATURE_DECIMAL_MODE` – 1 if decimal mode is enabled, 0 otherwise
* `CPUFEATURE_65C02` , `CPUFEATURE_65CE02` , `CPUFEATURE_HUC6280` , `CPUFEATURE_65816_EMULATION` , `CPUFEATURE_65816_NATIVE` ,
2019-06-12 12:06:02 +02:00
`CPUFEATURE_8080` , `CPUFEATURE_8085` , `CPUFEATURE_GAMEBOY` , `CPUFEATURE_Z80` ,
2019-10-01 00:46:15 +02:00
`CPUFEATURE_6502_ILLEGALS` , `CPUFEATURE_8085_ILLEGALS` , `CPUFEATURE_Z80_ILLEGALS` , `CPUFEATURE_Z80_NEXT` – 1 if given instruction subset is enabled, 0 otherwise
2018-08-08 13:44:30 +02:00
2020-03-31 19:07:35 +02:00
* `ENCCONV_SUPPORTED` - 1 if the module `encconv` supports the function `to_screencode` and other related functions, 0 otherwise.
2019-11-04 02:28:12 +01:00
2019-09-04 01:14:14 +02:00
* `ENCODING_SAME` - 1 if the encodings `default` and `src` are the same, 0 otherwise.
2020-03-31 19:07:35 +02:00
* `DECIMALS_SAME` - 1 if the encodings `default` and `src` have the same string terminator and decimal digits `'0'` -`'9'` , 0 otherwise.
2019-10-31 12:20:20 +01:00
* `NULLCHAR_SAME` - 1 if the encodings `default` and `src` have the same string terminator, 0 otherwise.
* `NULLCHAR` – the value of the `nullchar` constant
* `NULLCHAR_SRC` – the value of the `nullchar_src` constant
2019-06-28 16:28:49 +02:00
* `INIT_RW_MEMORY` – 1 if the option `ram_init_segment` is defined, 0 otherwise.
See [the ROM vs RAM guide ](../api/rom-vs-ram.md ) for more information.
2019-07-15 14:15:38 +02:00
* `BIG_ENDIAN` – 1 if the platform is big-endian, 0 otherwise (currently all supported platforms are little-endian)
* `LITTLE_ENDIAN` – 1 if the platform is little-endian, 0 otherwise (currently all supported platforms are little-endian)
2018-08-08 13:44:30 +02:00
* `OPTIMIZE_FOR_SIZE` , `OPTIMIZE_FOR_SPEED` , `OPTIMIZE_INLINE` , `OPTIMIZE_IPO`
– 1 if given optimization setting is enabled, 0 otherwise
2018-07-30 18:16:50 +02:00
2018-08-08 13:44:30 +02:00
* `SYNTAX_INTEL` , `SYNTAX_ZILOG` – 1 if given assembly syntax is chosen, 0 otherwise; doesn't take this file's pragmas into account
2018-07-30 18:16:50 +02:00
2018-08-08 13:44:30 +02:00
* `USES_ZPREG` – 1 if the zeropage pseudoregister is used, 0 otherwise
2018-07-30 18:16:50 +02:00
2019-06-12 12:06:02 +02:00
* `ZPREG_SIZE` – size of the pseudoregister in bytes, or 0 on platforms that don't use it
2018-08-08 13:44:30 +02:00
2019-06-28 16:44:41 +02:00
* `TINY_RW_MEMORY` – 1 if the main ram is 256 bytes or less, 0 otherwise
2019-06-28 16:28:49 +02:00
2018-08-08 13:44:30 +02:00
* `USES_IX_STACK` , `USES_IY_STACK` – 1 if given index register is used as a base pointer for stack-allocated variables, 0 otherwise
2019-06-12 12:06:02 +02:00
* `USES_SHADOW_REGISTERS` – 1 if interrupts preserve old registers in the shadow registers, 0 if they do it on stack
2018-07-12 18:30:35 +02:00
2019-06-12 12:06:02 +02:00
* `USES_SOFTWARE_STACK` – 1 if using software stack for variables (6502-like targets only), 0 otherwise
2018-12-14 22:01:52 +01:00
2018-07-12 18:30:35 +02:00
### Commonly used features
2020-03-31 19:07:35 +02:00
These features are frequently defined in the platform definition file.
Some libraries may require that some of these be defined.
2018-12-17 11:13:01 +01:00
2020-03-31 19:07:35 +02:00
* `WIDESCREEN` – 1 if the horizontal screen resolution, ignoring borders, is greater than 256, 0 otherwise
2018-08-08 13:44:30 +02:00
2018-07-12 18:30:35 +02:00
* `KEYBOARD` – 1 if the target has a keyboard, 0 otherwise
* `JOYSTICKS` – the maximum number of joysticks using standard hardware configurations, may be 0
* `HAS_BITMAP_MODE` – 1 if the target has a display mode with every pixel addressable, 0 otherwise
2020-03-31 19:07:35 +02:00
* `NTSC` – 1 if the target is NTSC, 0 otherwise
2018-08-08 13:44:30 +02:00
2020-03-31 19:07:35 +02:00
* `PAL` – 1 if the target is PAL, 0 otherwise
2018-08-08 13:44:30 +02:00
2020-03-31 19:07:35 +02:00
* `NULLPTR` – physical value of `nullptr` , default 0
2019-05-31 17:03:35 +02:00
2020-03-31 19:07:35 +02:00
### Target-specific features
2019-06-05 13:31:43 +02:00
2020-03-31 19:07:35 +02:00
These features are used to identify the target machine in multiplatform programs and libraries:
2018-12-19 19:01:53 +01:00
2020-03-31 19:07:35 +02:00
* `CBM` – 1 if the target is an 8-bit Commodore computer (or a compatible one), 0 otherwise
(for more Commodore-related preprocessor options, see [Preprocessor options for Commodore computer targets ](./preprocessor_cbm.md ))
2018-12-19 19:01:53 +01:00
2020-03-31 19:07:35 +02:00
* `AMSTRAD_CPC` , `ATARI_2600` , `ATARI_8` , `ATARI_LYNX` , `APPLE_2` , `BBC_MICRO` ,
`COMMANDER_X16` , `CPM` , `GAMEBOY` , `IBM_PC` , `MSX` , `NEC_PC_88` , `NES` , `ZX_SPECTRUM`
– 1 if the target is the machine in question, 0 otherwise
2019-04-16 16:56:23 +02:00
2019-09-16 22:55:08 +02:00
* `VERA_VERSION` – on Commander X16, the version of the VERA chip: `7` for 0.7, `8` for 0.8
2020-03-31 19:07:35 +02:00
2018-07-12 18:30:35 +02:00
### Built-in preprocessor functions and operators
The `defined` function returns 1 if the feature is defined, 0 otherwise.
All the other functions and operators treat undefined features as if they were defined as 0.
2019-06-14 12:00:45 +02:00
The `if` function returns its second parameter if the first parameter is defined and non-zero, and the third parameter otherwise:
// prints 400:
#infoeval if(1, 400, 500)
// prints 500:
#infoeval if(0, 400, 500)
TODO
2020-03-19 19:43:24 +01:00
`not` , `lo` , `hi` , `min` , `max` `+` , `-` , `*` , `|` , `&` , `^` , `||` , `&&` , `<<` , `>>` ,`==` , `!=` , `>` , `>=` , `<` , `<=`
2018-07-12 18:30:35 +02:00
The following Millfork operators and functions are not available in the preprocessor:
`+'` , `-'` , `*'` , `<<'` , `>>'` , `:` , `>>>>` , `nonet` , all the assignment operators
### `#if/#elseif/#else/#endif`
#if < expr >
#elseif < expr >
#else
#endif
TODO
### `#fatal/#error/#warn/#info`
2019-09-16 22:48:21 +02:00
#fatal fatal error message
2018-07-12 18:30:35 +02:00
#error error message
#warn warning message
#info informational message
Emits a diagnostic message.
`#fatal` interrupts the compilation immediately.
`#error` causes an error, but the compilation will continue.
`#warn` emits a warning. It may be treated as an error depending on compilation options.
`#info` emits a benign diagnostic message.
2018-09-28 22:41:57 +02:00
### `#infoeval`
#infoeval < expr >
Evaluates an expression and emits the result as a diagnostic message.
2018-07-12 18:30:35 +02:00
2019-06-26 13:43:15 +02:00
### `#define`
#define < ident > = < expr >
Defines a new feature value or redefines a previous feature value.
The feature value is visible only to the preprocessor, only when processing the current file,
and only in lines preprocessed after this one.
2018-07-12 18:30:35 +02:00
### `#use`
2019-06-14 12:00:45 +02:00
#use < ident > = < expr >
#use < feature >
// equivalent to #use < feature > = < feature >
Exports a value to the parser.
The parser will substitute every use of the given identifier as a variable or constant
2018-07-12 18:30:35 +02:00
with the numeric value of the feature.
2019-06-14 12:00:45 +02:00
#use CPU_MODEL = if(CPU_65816 | ARCH_X86, 16, 8)
putstrz("Your CPU is "z)
putword(CPU_MODEL)
putstrz("-bit."z)
2018-07-12 18:30:35 +02:00
The substitution will happen only within the current file.
2019-06-14 12:00:45 +02:00
To use such value in other files, consider using a normal constant:
2018-07-12 18:30:35 +02:00
#use WIDESCREEN
const byte is_widescreen = WIDESCREEN
2018-08-03 13:23:37 +02:00
### `#pragma`
Changes the behaviour of the parser for the current file.
2019-06-12 12:06:02 +02:00
The change applies to the whole file, regardless of where the directive is located.
2018-08-03 13:23:37 +02:00
* `#pragma intel_syntax` – interpret assembly using Intel syntax
* `#pragma zilog_syntax` – interpret assembly using Zilog syntax
2018-07-12 18:30:35 +02:00