avr: properly support the .rodata section

Unfortunately, the .rodata section can't be stored in flash. Instead, an
explicit .progmem section should be used, which is supported in LLVM as
address space 1 but not exposed to normal programs.

Eventually a pass should be written that converts trivial const globals
of which all loads are visible to be in addrspace 1, to get the benefits
of storing those globals directly in ROM.
Этот коммит содержится в:
Ayke van Laethem 2020-10-31 20:40:20 +01:00 коммит произвёл Ron Evans
родитель 3364da6f25
коммит 171f793c1e
6 изменённых файлов: 4 добавлений и 44 удалений

Просмотреть файл

@ -117,18 +117,6 @@ func Build(pkgName, outpath string, config *compileopts.Config, action func(Buil
return errors.New("verification failure after LLVM optimization passes")
}
// On the AVR, pointers can point either to flash or to RAM, but we don't
// know. As a temporary fix, load all global variables in RAM.
// In the future, there should be a compiler pass that determines which
// pointers are flash and which are in RAM so that pointers can have a
// correct address space parameter (address space 1 is for flash).
if strings.HasPrefix(config.Triple(), "avr") {
transform.NonConstGlobals(mod)
if err := llvm.VerifyModule(mod, llvm.PrintMessageAction); err != nil {
return errors.New("verification error after making all globals non-constant on AVR")
}
}
// LLVM 11 by default tries to emit tail calls (even with the target feature
// disabled) unless it is explicitly disabled with a function attribute.
// This is a problem, as it tries to emit them and prints an error when it

Просмотреть файл

@ -13,8 +13,8 @@ SECTIONS
KEEP(*(.text.__vector_RESET))
KEEP(*(.text.main)) /* main must follow the reset handler */
*(.text.*)
*(.rodata)
*(.rodata.*)
*(.progmem)
*(.progmem.*)
}
.stack (NOLOAD) :
@ -28,6 +28,8 @@ SECTIONS
.data :
{
_sdata = .; /* used by startup code */
*(.rodata)
*(.rodata.*)
*(.data)
*(.data*)
_edata = .; /* used by startup code */

Просмотреть файл

@ -19,19 +19,6 @@ func ApplyFunctionSections(mod llvm.Module) {
}
}
// NonConstGlobals turns all global constants into global variables. This works
// around a limitation on Harvard architectures (e.g. AVR), where constant and
// non-constant pointers point to a different address space. Normal pointer
// behavior is restored by using the data space only, at the cost of RAM for
// constant global variables.
func NonConstGlobals(mod llvm.Module) {
global := mod.FirstGlobal()
for !global.IsNil() {
global.SetGlobalConstant(false)
global = llvm.NextGlobal(global)
}
}
// DisableTailCalls adds the "disable-tail-calls"="true" function attribute to
// all functions. This may be necessary, in particular to avoid an error with
// WebAssembly in LLVM 11.

Просмотреть файл

@ -12,10 +12,3 @@ func TestApplyFunctionSections(t *testing.T) {
ApplyFunctionSections(mod)
})
}
func TestNonConstGlobals(t *testing.T) {
t.Parallel()
testTransform(t, "testdata/globals-non-const", func(mod llvm.Module) {
NonConstGlobals(mod)
})
}

5
transform/testdata/globals-non-const.ll предоставленный
Просмотреть файл

@ -1,5 +0,0 @@
target datalayout = "e-m:e-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S64"
target triple = "armv7em-none-eabi"
@globalIntConst = constant i32 3
@globalIntVar = global i32 5

5
transform/testdata/globals-non-const.out.ll предоставленный
Просмотреть файл

@ -1,5 +0,0 @@
target datalayout = "e-m:e-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S64"
target triple = "armv7em-none-eabi"
@globalIntConst = global i32 3
@globalIntVar = global i32 5