Using TCF
General sescription
Currently GNU toolchain has a partial support for TCF, however it is not complete and in particular scenarios TCFs cannot be used as-is.
If you are using Eclipse IDE for ARC, please refer to a Building User Guide. Eclipse IDE for ARC supports only GCC compiler and GNU linker script sections of TCF, it doesn’t support preprocessor defines sections as of version 2016.03.
If you are using GNU toolchain without IDE on Linux hosts you can use a special
script arc-elf32-tcf-gcc (for big-endian toolchain this file has
arceb- prefix) that is located in the same bin
directory as rest
of the toolchain executable files. This executable accepts all of the same
options as GCC driver and also an option --tcf <PATH/TO/TCF>
.
arc-elf32-tcf-gcc will extract compiler options, linker script and
preprocessor defines from TCF and will pass them to GCC along with other
options.
GCC options from
gcc_compiler
section will be passed as-is, but can be overridden by-m<something>
options passed directly to arc-elf32-tcf-gcc.GNU linker script will be extracted from
gnu_linker_command_file
will be used as amemory.x
file for-Wl,marcv2elfx
linker emulation. Option-Wl,-marcv2elfx
is added by this wrapper - there is no need to pass it explicitly.Preprocessor defines from section
C_defines
will be passed with-include
option of GCC.
arc-elf32-tcf-gcc is a Perl script that requires XML:LibXML
package. It is likely to work on most Linux hosts, however it will not work on
Windows hosts, unless Perl with required library has been installed and added
to the PATH
environment variable. TCF is a text file in XML format, so in
case of need it is trivial to extract compiler flags and linker script from TCF
and use them directly with GCC and ld without IDE or wrapper script.
Value of -mcpu=
option is selected by TCF generator to have best match with
the target processor. This option ARC Toolchain Variants not only sets various
hardware options but also selects a particular build of standard library.
Values of hardware extensions can be overridden with individual -m*
options, but that will not change standard library to a matching build - it
still will use standard library build selected by -mcpu=
value.
Compiler options
GCC options are stored in the gcc_compiler
section of TCF. These options
are passed to GCC as-is. These are “machine-specific” options applicable only
to ARC, and which define configuration of target architecture - which of the
optional hardware extensions (like bitscan instructions, barrel shifter
instructions, etc) are present. Application that uses hardware extensions will
not work on ARC processor without those extensions - there will be an Illegal
instruction exception (although application may emulate instruction via
handling of this exception, but that is out of scope of this document).
Application that doesn’t use hardware extensions present in the target ARC
processor might be ineffective, if those extensions allow more optimal
implementation of same algorithm. Usually hardware extensions allow improvement
of both code size and performance at the expense of increased gate count, with
all respective consequences.
When TCF is selected in the IDE respective compiler options are disabled in GUI and cannot be changed by user. However if TCF is deselected those options remain at selected values, so it is possible to “import” options from TCF and then modify it for particular need.
When using arc-elf32-tcf-gcc compiler options passed to this wrapper script has a higher precedence then options in TCF, so it is possible to use TCF as a “baseline” and then modify if needed.
Memory map
Please refer to main page about GNU linker for ARC Linker scripts and memory.x files for more details.
TCF doesn’t contain a linker script for GNU linker in the strict meaning of
this term. Instead TCF contains a special memory map, which can be used
together with a linker emulation called arcv2elfx. This linker emulation
reads a special file called memory.x
to get several defines which denote
location of particular memory areas, and then emulation allocates ELF sections
to those areas. So, for example, memory.x
may specify address and size of
ICCM and DCCM memories and linker would put code sections into ICCM and data
sections to DCCM. TCF contains this memory.x
file as content of
gnu_linker_command_file
section. IDE and arc-elf32-tcf-gcc simply create
this file and specify to linker to use arcv2elfx emulation. This is done by
passing option -marcv2elfx
to linker, but note that when invoking gcc
driver it is required to specify this option as -Wl,-marcv2elfx
, so driver
would know that this is an option to pass to linker.
It is very important that memory map in TCF matches the one in the hardware, otherwise application will not work. By default linker places all application code and data as a continuous sections starting from address 0x0. Designs with CCMs usually has ICCM mapped at address 0x0, and DCCM at addresses >= 0x8000_0000 (or simply an upper half of address space, which can be less then 32 bits wide). If application has both code and data put into ICCM, it may technically work (load/store unit in ARC has a port to ICCM), however this underutilizes DCCM and creates a risk of memory overflow where code and data will not fit into the ICCM, so overflown content will be lost, likely causing an error message in simulator or in debugger. For this reason it is recommended to use memory.x file from TCF when linking applications that use CCM memory. Typically TCF-generator would automatically assign instruction memory area to ICCM and data memory area to DCCM, because parameters of those memories can be read from BCRs, although it doesn’t support such features as ICCM1 or NV ICCM.
When memory is connected via external memory bus TCF-generator cannot know where memory will be actually located, so it will put all sections continuously, starting from address 0. This is basically same as what happens when no memory map has been passed to linker. Therefore memory map in such TCF is effectively useless, instead it is needed to manually enter a proper memory map into “gnu_linker_command_file” section. However when using an nSIM simulator such TCF will work nice, as it will make nSIM simulate whole address space, so there is no risk that application will be loaded into unexisting address.
When using IDE there is an option to ignore memory map specified in TCF and use default memory mapping or custom linker script. This is the default setting - to ignore linker script embedded into TCF. However if target design uses closely-coupled memories then it is highly advised to use memory map (embedded into TCF or manually written).
C preprocessor defines
TCF section C_defines
contains preprocessor defines that specify presence of
various hardware optional extensions and values of Build Configuration
Registers. arc-elf32-tcf-gcc wrapper extracts content of this section into
temporary file and includes into compiled files via -include
option of GCC.
arc-elf32-tcf-gcc options
- --compiler
Overwrites the default compiler name. The compiler tool chain needs to be in the PATH. Default value depends on the name of this file - it will call compiler that has the same name, only without -tcf part. Therefore:
arc-elf32-tcf-gcc -> arc-elf32-gcc
arceb-elf32-tcf-gcc -> arceb-elf32-gcc
arc-linux-tcf-gcc -> arc-linux-gcc
arceb-linux-tcf-gcc -> arceb-linux-gcc
arc-a-b-tcf-gcc -> arc-a-b-gcc
arc-tcf-elf32-tcf-gcc -> arc-tcf-elf32-gcc
- --tcf
The name and the location of the TCF file.
- --verbose
Verbose output. Prints the compiler invokation command.