all: use less magic in memory-mapped IO
Don't store addresses in the values of registers, this leads to problems with char arrays (among others). Instead, do it like it's done in C with raw addresses cast to struct pointers. This commit also splits gen-device.py, as AVR and ARM have very different ideas of what a register is. It's easier to just keep them separate.
Этот коммит содержится в:
родитель
93248c93ed
коммит
17b5b6ec5b
7 изменённых файлов: 296 добавлений и 209 удалений
4
Makefile
4
Makefile
|
@ -80,11 +80,11 @@ fmt:
|
||||||
gen-device: gen-device-nrf
|
gen-device: gen-device-nrf
|
||||||
|
|
||||||
gen-device-nrf:
|
gen-device-nrf:
|
||||||
./gen-device.py lib/nrfx/mdk/ src/device/nrf/
|
./gen-device-svd.py lib/nrfx/mdk/ src/device/nrf/
|
||||||
go fmt ./src/device/nrf
|
go fmt ./src/device/nrf
|
||||||
|
|
||||||
gen-device-avr:
|
gen-device-avr:
|
||||||
./gen-device.py lib/avr/packs/atmega src/device/avr/
|
./gen-device-avr.py lib/avr/packs/atmega src/device/avr/
|
||||||
go fmt ./src/device/avr
|
go fmt ./src/device/avr
|
||||||
|
|
||||||
|
|
||||||
|
|
23
compiler.go
23
compiler.go
|
@ -1421,15 +1421,12 @@ func (c *Compiler) parseInstr(frame *Frame, instr ssa.Instruction) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
store := c.builder.CreateStore(llvmVal, llvmAddr)
|
||||||
valType := instr.Addr.Type().(*types.Pointer).Elem()
|
valType := instr.Addr.Type().(*types.Pointer).Elem()
|
||||||
if valType, ok := valType.(*types.Named); ok && valType.Obj().Name() == "__reg" {
|
if valType, ok := valType.(*types.Named); ok && valType.Obj().Name() == "__reg" {
|
||||||
// Magic type name to transform this store to a register store.
|
// Magic type name to make this store volatile, for memory-mapped
|
||||||
registerAddr := c.builder.CreateLoad(llvmAddr, "")
|
// registers.
|
||||||
ptr := c.builder.CreateIntToPtr(registerAddr, llvmAddr.Type(), "")
|
|
||||||
store := c.builder.CreateStore(llvmVal, ptr)
|
|
||||||
store.SetVolatile(true)
|
store.SetVolatile(true)
|
||||||
} else {
|
|
||||||
c.builder.CreateStore(llvmVal, llvmAddr)
|
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
default:
|
default:
|
||||||
|
@ -2739,19 +2736,13 @@ func (c *Compiler) parseUnOp(frame *Frame, unop *ssa.UnOp) (llvm.Value, error) {
|
||||||
}
|
}
|
||||||
case token.MUL: // *x, dereference pointer
|
case token.MUL: // *x, dereference pointer
|
||||||
valType := unop.X.Type().(*types.Pointer).Elem()
|
valType := unop.X.Type().(*types.Pointer).Elem()
|
||||||
|
load := c.builder.CreateLoad(x, "")
|
||||||
if valType, ok := valType.(*types.Named); ok && valType.Obj().Name() == "__reg" {
|
if valType, ok := valType.(*types.Named); ok && valType.Obj().Name() == "__reg" {
|
||||||
// Magic type name: treat the value as a register pointer.
|
// Magic type name to make this load volatile, for memory-mapped
|
||||||
register := unop.X.(*ssa.FieldAddr)
|
// registers.
|
||||||
global := register.X.(*ssa.Global)
|
|
||||||
llvmGlobal := c.ir.GetGlobal(global).llvmGlobal
|
|
||||||
llvmAddr := c.builder.CreateExtractValue(llvmGlobal.Initializer(), register.Field, "")
|
|
||||||
ptr := llvm.ConstIntToPtr(llvmAddr, x.Type())
|
|
||||||
load := c.builder.CreateLoad(ptr, "")
|
|
||||||
load.SetVolatile(true)
|
load.SetVolatile(true)
|
||||||
return load, nil
|
|
||||||
} else {
|
|
||||||
return c.builder.CreateLoad(x, ""), nil
|
|
||||||
}
|
}
|
||||||
|
return load, nil
|
||||||
case token.XOR: // ^x, toggle all bits in integer
|
case token.XOR: // ^x, toggle all bits in integer
|
||||||
return c.builder.CreateXor(x, llvm.ConstInt(x.Type(), ^uint64(0), false), ""), nil
|
return c.builder.CreateXor(x, llvm.ConstInt(x.Type(), ^uint64(0), false), ""), nil
|
||||||
default:
|
default:
|
||||||
|
|
217
gen-device-avr.py
Исполняемый файл
217
gen-device-avr.py
Исполняемый файл
|
@ -0,0 +1,217 @@
|
||||||
|
#!/usr/bin/python3
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
from xml.dom import minidom
|
||||||
|
from glob import glob
|
||||||
|
from collections import OrderedDict
|
||||||
|
import re
|
||||||
|
|
||||||
|
class Device:
|
||||||
|
# dummy
|
||||||
|
pass
|
||||||
|
|
||||||
|
def getText(element):
|
||||||
|
strings = []
|
||||||
|
for node in element.childNodes:
|
||||||
|
if node.nodeType == node.TEXT_NODE:
|
||||||
|
strings.append(node.data)
|
||||||
|
return ''.join(strings)
|
||||||
|
|
||||||
|
def formatText(text):
|
||||||
|
text = re.sub('[ \t\n]+', ' ', text) # Collapse whitespace (like in HTML)
|
||||||
|
text = text.replace('\\n ', '\n')
|
||||||
|
text = text.strip()
|
||||||
|
return text
|
||||||
|
|
||||||
|
def readATDF(path):
|
||||||
|
# Read Atmel device descriptor files.
|
||||||
|
# See: http://packs.download.atmel.com
|
||||||
|
|
||||||
|
device = Device()
|
||||||
|
|
||||||
|
xml = minidom.parse(path)
|
||||||
|
device = xml.getElementsByTagName('device')[0]
|
||||||
|
deviceName = device.getAttribute('name')
|
||||||
|
arch = device.getAttribute('architecture')
|
||||||
|
family = device.getAttribute('family')
|
||||||
|
|
||||||
|
memorySizes = {}
|
||||||
|
for el in device.getElementsByTagName('address-space'):
|
||||||
|
addressSpace = {
|
||||||
|
'size': int(el.getAttribute('size'), 0),
|
||||||
|
'segments': {},
|
||||||
|
}
|
||||||
|
memorySizes[el.getAttribute('name')] = addressSpace
|
||||||
|
for segmentEl in el.getElementsByTagName('memory-segment'):
|
||||||
|
addressSpace['segments'][segmentEl.getAttribute('name')] = int(segmentEl.getAttribute('size'), 0)
|
||||||
|
|
||||||
|
device.interrupts = []
|
||||||
|
for el in device.getElementsByTagName('interrupts')[0].getElementsByTagName('interrupt'):
|
||||||
|
device.interrupts.append({
|
||||||
|
'index': int(el.getAttribute('index')),
|
||||||
|
'name': el.getAttribute('name'),
|
||||||
|
'description': el.getAttribute('caption'),
|
||||||
|
})
|
||||||
|
|
||||||
|
allRegisters = {}
|
||||||
|
commonRegisters = {}
|
||||||
|
|
||||||
|
device.peripherals = []
|
||||||
|
for el in xml.getElementsByTagName('modules')[0].getElementsByTagName('module'):
|
||||||
|
peripheral = {
|
||||||
|
'name': el.getAttribute('name'),
|
||||||
|
'description': el.getAttribute('caption'),
|
||||||
|
'registers': [],
|
||||||
|
}
|
||||||
|
device.peripherals.append(peripheral)
|
||||||
|
for regElGroup in el.getElementsByTagName('register-group'):
|
||||||
|
for regEl in regElGroup.getElementsByTagName('register'):
|
||||||
|
size = int(regEl.getAttribute('size'))
|
||||||
|
regName = regEl.getAttribute('name')
|
||||||
|
regOffset = int(regEl.getAttribute('offset'), 0)
|
||||||
|
reg = {
|
||||||
|
'description': regEl.getAttribute('caption'),
|
||||||
|
'bitfields': [],
|
||||||
|
'array': None,
|
||||||
|
}
|
||||||
|
if size == 1:
|
||||||
|
reg['variants'] = [{
|
||||||
|
'name': regName,
|
||||||
|
'address': regOffset,
|
||||||
|
}]
|
||||||
|
elif size == 2:
|
||||||
|
reg['variants'] = [{
|
||||||
|
'name': regName + 'L',
|
||||||
|
'address': regOffset,
|
||||||
|
}, {
|
||||||
|
'name': regName + 'H',
|
||||||
|
'address': regOffset + 1,
|
||||||
|
}]
|
||||||
|
else:
|
||||||
|
# TODO
|
||||||
|
continue
|
||||||
|
|
||||||
|
for bitfieldEl in regEl.getElementsByTagName('bitfield'):
|
||||||
|
reg['bitfields'].append({
|
||||||
|
'name': regName + '_' + bitfieldEl.getAttribute('name'),
|
||||||
|
'description': bitfieldEl.getAttribute('caption'),
|
||||||
|
'value': int(bitfieldEl.getAttribute('mask'), 0),
|
||||||
|
})
|
||||||
|
|
||||||
|
if regName in allRegisters:
|
||||||
|
firstReg = allRegisters[regName]
|
||||||
|
if firstReg['register'] in firstReg['peripheral']['registers']:
|
||||||
|
firstReg['peripheral']['registers'].remove(firstReg['register'])
|
||||||
|
if firstReg['address'] != regOffset:
|
||||||
|
continue # TODO
|
||||||
|
commonRegisters = allRegisters[regName]['register']
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
allRegisters[regName] = {'address': regOffset, 'register': reg, 'peripheral': peripheral}
|
||||||
|
|
||||||
|
peripheral['registers'].append(reg)
|
||||||
|
|
||||||
|
device.metadata = {
|
||||||
|
'file': os.path.basename(path),
|
||||||
|
'descriptorSource': 'http://packs.download.atmel.com/',
|
||||||
|
'name': deviceName,
|
||||||
|
'nameLower': deviceName.lower(),
|
||||||
|
'description': 'Device information for the {}.'.format(deviceName),
|
||||||
|
'arch': arch,
|
||||||
|
'family': family,
|
||||||
|
'flashSize': memorySizes['prog']['size'],
|
||||||
|
'ramSize': memorySizes['data']['segments'].get('IRAM', memorySizes['data']['segments'].get('INTERNAL_SRAM')),
|
||||||
|
'numInterrupts': len(device.interrupts),
|
||||||
|
}
|
||||||
|
|
||||||
|
return device
|
||||||
|
|
||||||
|
def writeGo(outdir, device):
|
||||||
|
# The Go module for this device.
|
||||||
|
out = open(outdir + '/' + device.metadata['nameLower'] + '.go', 'w')
|
||||||
|
pkgName = os.path.basename(outdir.rstrip('/'))
|
||||||
|
out.write('''\
|
||||||
|
// Automatically generated file. DO NOT EDIT.
|
||||||
|
// Generated by gen-device-avr.py from {file}, see {descriptorSource}
|
||||||
|
|
||||||
|
// +build {pkgName},{nameLower}
|
||||||
|
|
||||||
|
// {description}
|
||||||
|
package {pkgName}
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
// Magic type name for the compiler.
|
||||||
|
type __reg uint8
|
||||||
|
|
||||||
|
// Export this magic type name.
|
||||||
|
type RegValue = __reg
|
||||||
|
|
||||||
|
// Some information about this device.
|
||||||
|
const (
|
||||||
|
DEVICE = "{name}"
|
||||||
|
ARCH = "{arch}"
|
||||||
|
FAMILY = "{family}"
|
||||||
|
)
|
||||||
|
'''.format(pkgName=pkgName, **device.metadata))
|
||||||
|
|
||||||
|
out.write('\n// Interrupts\nconst (\n')
|
||||||
|
for intr in device.interrupts:
|
||||||
|
out.write('\tIRQ_{name} = {index} // {description}\n'.format(**intr))
|
||||||
|
intrMax = max(map(lambda intr: intr['index'], device.interrupts))
|
||||||
|
out.write('\tIRQ_max = {} // Highest interrupt number on this device.\n'.format(intrMax))
|
||||||
|
out.write(')\n')
|
||||||
|
|
||||||
|
out.write('\n// Peripherals.\nvar (')
|
||||||
|
first = True
|
||||||
|
for peripheral in device.peripherals:
|
||||||
|
out.write('\n\t// {description}\n'.format(**peripheral))
|
||||||
|
for register in peripheral['registers']:
|
||||||
|
for variant in register['variants']:
|
||||||
|
out.write('\t{name} = (*__reg)(unsafe.Pointer(uintptr(0x{address:x})))\n'.format(**variant))
|
||||||
|
out.write(')\n')
|
||||||
|
|
||||||
|
for peripheral in device.peripherals:
|
||||||
|
if not sum(map(lambda r: len(r['bitfields']), peripheral['registers'])): continue
|
||||||
|
out.write('\n// Bitfields for {name}: {description}\nconst('.format(**peripheral))
|
||||||
|
for register in peripheral['registers']:
|
||||||
|
if not register['bitfields']: continue
|
||||||
|
for variant in register['variants']:
|
||||||
|
out.write('\n\t// {name}'.format(**variant))
|
||||||
|
if register['description']:
|
||||||
|
out.write(': {description}'.format(**register))
|
||||||
|
out.write('\n')
|
||||||
|
for bitfield in register['bitfields']:
|
||||||
|
out.write('\t{name} = 0x{value:x}'.format(**bitfield))
|
||||||
|
if bitfield['description']:
|
||||||
|
out.write(' // {description}'.format(**bitfield))
|
||||||
|
out.write('\n')
|
||||||
|
out.write(')\n')
|
||||||
|
|
||||||
|
def writeLD(outdir, device):
|
||||||
|
# Variables for the linker script.
|
||||||
|
out = open(outdir + '/' + device.metadata['nameLower'] + '.ld', 'w')
|
||||||
|
out.write('''\
|
||||||
|
/* Automatically generated file. DO NOT EDIT. */
|
||||||
|
/* Generated by gen-device-avr.py from {file}, see {descriptorSource} */
|
||||||
|
|
||||||
|
__flash_size = 0x{flashSize:x};
|
||||||
|
__ram_size = 0x{ramSize:x};
|
||||||
|
__num_isrs = {numInterrupts};
|
||||||
|
'''.format(**device.metadata))
|
||||||
|
out.close()
|
||||||
|
|
||||||
|
|
||||||
|
def generate(indir, outdir):
|
||||||
|
for filepath in sorted(glob(indir + '/*.atdf')):
|
||||||
|
print(filepath)
|
||||||
|
device = readATDF(filepath)
|
||||||
|
writeGo(outdir, device)
|
||||||
|
writeLD(outdir, device)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
indir = sys.argv[1] # directory with register descriptor files (*.atdf)
|
||||||
|
outdir = sys.argv[2] # output directory
|
||||||
|
generate(indir, outdir)
|
|
@ -52,6 +52,7 @@ def readSVD(path):
|
||||||
peripheral = {
|
peripheral = {
|
||||||
'name': name,
|
'name': name,
|
||||||
'description': description,
|
'description': description,
|
||||||
|
'baseAddress': baseAddress,
|
||||||
'registers': [],
|
'registers': [],
|
||||||
}
|
}
|
||||||
device.peripherals.append(peripheral)
|
device.peripherals.append(peripheral)
|
||||||
|
@ -94,7 +95,6 @@ def readSVD(path):
|
||||||
'nameLower': deviceName.lower(),
|
'nameLower': deviceName.lower(),
|
||||||
'description': deviceDescription,
|
'description': deviceDescription,
|
||||||
'licenseBlock': '\n// ' + licenseText.replace('\n', '\n// '),
|
'licenseBlock': '\n// ' + licenseText.replace('\n', '\n// '),
|
||||||
'regType': 'uint32',
|
|
||||||
'arch': ARM_ARCHS[cpuName],
|
'arch': ARM_ARCHS[cpuName],
|
||||||
'family': getText(root.getElementsByTagName('series')[0]),
|
'family': getText(root.getElementsByTagName('series')[0]),
|
||||||
}
|
}
|
||||||
|
@ -134,119 +134,13 @@ def parseSVDRegister(peripheralName, regEl, baseAddress, namePrefix=''):
|
||||||
})
|
})
|
||||||
|
|
||||||
return {
|
return {
|
||||||
'variants': [{
|
'name': namePrefix + regName,
|
||||||
'name': namePrefix + regName,
|
'address': address,
|
||||||
'address': address,
|
|
||||||
}],
|
|
||||||
'description': regDescription.replace('\n', ' '),
|
'description': regDescription.replace('\n', ' '),
|
||||||
'bitfields': fields,
|
'bitfields': fields,
|
||||||
'array': array,
|
'array': array,
|
||||||
}
|
}
|
||||||
|
|
||||||
def readATDF(path):
|
|
||||||
# Read Atmel device descriptor files.
|
|
||||||
# See: http://packs.download.atmel.com
|
|
||||||
|
|
||||||
device = Device()
|
|
||||||
|
|
||||||
xml = minidom.parse(path)
|
|
||||||
device = xml.getElementsByTagName('device')[0]
|
|
||||||
deviceName = device.getAttribute('name')
|
|
||||||
arch = device.getAttribute('architecture')
|
|
||||||
family = device.getAttribute('family')
|
|
||||||
|
|
||||||
memorySizes = {}
|
|
||||||
for el in device.getElementsByTagName('address-space'):
|
|
||||||
addressSpace = {
|
|
||||||
'size': int(el.getAttribute('size'), 0),
|
|
||||||
'segments': {},
|
|
||||||
}
|
|
||||||
memorySizes[el.getAttribute('name')] = addressSpace
|
|
||||||
for segmentEl in el.getElementsByTagName('memory-segment'):
|
|
||||||
addressSpace['segments'][segmentEl.getAttribute('name')] = int(segmentEl.getAttribute('size'), 0)
|
|
||||||
|
|
||||||
device.interrupts = []
|
|
||||||
for el in device.getElementsByTagName('interrupts')[0].getElementsByTagName('interrupt'):
|
|
||||||
device.interrupts.append({
|
|
||||||
'index': int(el.getAttribute('index')),
|
|
||||||
'name': el.getAttribute('name'),
|
|
||||||
'description': el.getAttribute('caption'),
|
|
||||||
})
|
|
||||||
|
|
||||||
allRegisters = {}
|
|
||||||
commonRegisters = {}
|
|
||||||
|
|
||||||
device.peripherals = []
|
|
||||||
for el in xml.getElementsByTagName('modules')[0].getElementsByTagName('module'):
|
|
||||||
peripheral = {
|
|
||||||
'name': el.getAttribute('name'),
|
|
||||||
'description': el.getAttribute('caption'),
|
|
||||||
'registers': [],
|
|
||||||
}
|
|
||||||
device.peripherals.append(peripheral)
|
|
||||||
for regElGroup in el.getElementsByTagName('register-group'):
|
|
||||||
for regEl in regElGroup.getElementsByTagName('register'):
|
|
||||||
size = int(regEl.getAttribute('size'))
|
|
||||||
regName = regEl.getAttribute('name')
|
|
||||||
regOffset = int(regEl.getAttribute('offset'), 0)
|
|
||||||
reg = {
|
|
||||||
'description': regEl.getAttribute('caption'),
|
|
||||||
'bitfields': [],
|
|
||||||
'array': None,
|
|
||||||
}
|
|
||||||
if size == 1:
|
|
||||||
reg['variants'] = [{
|
|
||||||
'name': regName,
|
|
||||||
'address': regOffset,
|
|
||||||
}]
|
|
||||||
elif size == 2:
|
|
||||||
reg['variants'] = [{
|
|
||||||
'name': regName + 'L',
|
|
||||||
'address': regOffset,
|
|
||||||
}, {
|
|
||||||
'name': regName + 'H',
|
|
||||||
'address': regOffset,
|
|
||||||
}]
|
|
||||||
else:
|
|
||||||
reg['variants'] = [] # TODO
|
|
||||||
|
|
||||||
for bitfieldEl in regEl.getElementsByTagName('bitfield'):
|
|
||||||
reg['bitfields'].append({
|
|
||||||
'name': regName + '_' + bitfieldEl.getAttribute('name'),
|
|
||||||
'description': bitfieldEl.getAttribute('caption'),
|
|
||||||
'value': int(bitfieldEl.getAttribute('mask'), 0),
|
|
||||||
})
|
|
||||||
|
|
||||||
if regName in allRegisters:
|
|
||||||
firstReg = allRegisters[regName]
|
|
||||||
if firstReg['register'] in firstReg['peripheral']['registers']:
|
|
||||||
firstReg['peripheral']['registers'].remove(firstReg['register'])
|
|
||||||
if firstReg['address'] != regOffset:
|
|
||||||
continue # TODO
|
|
||||||
commonRegisters = allRegisters[regName]['register']
|
|
||||||
continue
|
|
||||||
else:
|
|
||||||
allRegisters[regName] = {'address': regOffset, 'register': reg, 'peripheral': peripheral}
|
|
||||||
|
|
||||||
peripheral['registers'].append(reg)
|
|
||||||
|
|
||||||
device.metadata = {
|
|
||||||
'file': os.path.basename(path),
|
|
||||||
'descriptorSource': 'http://packs.download.atmel.com/',
|
|
||||||
'name': deviceName,
|
|
||||||
'nameLower': deviceName.lower(),
|
|
||||||
'description': 'Device information for the {}.'.format(deviceName),
|
|
||||||
'licenseBlock': '',
|
|
||||||
'regType': 'uint8',
|
|
||||||
'arch': arch,
|
|
||||||
'family': family,
|
|
||||||
'flashSize': memorySizes['prog']['size'],
|
|
||||||
'ramSize': memorySizes['data']['segments'].get('IRAM', memorySizes['data']['segments'].get('INTERNAL_SRAM')),
|
|
||||||
'numInterrupts': len(device.interrupts),
|
|
||||||
}
|
|
||||||
|
|
||||||
return device
|
|
||||||
|
|
||||||
def writeGo(outdir, device):
|
def writeGo(outdir, device):
|
||||||
# The Go module for this device.
|
# The Go module for this device.
|
||||||
out = open(outdir + '/' + device.metadata['nameLower'] + '.go', 'w')
|
out = open(outdir + '/' + device.metadata['nameLower'] + '.go', 'w')
|
||||||
|
@ -261,8 +155,10 @@ def writeGo(outdir, device):
|
||||||
// {licenseBlock}
|
// {licenseBlock}
|
||||||
package {pkgName}
|
package {pkgName}
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
// Magic type name for the compiler.
|
// Magic type name for the compiler.
|
||||||
type __reg {regType}
|
type __reg uint32
|
||||||
|
|
||||||
// Export this magic type name.
|
// Export this magic type name.
|
||||||
type RegValue = __reg
|
type RegValue = __reg
|
||||||
|
@ -282,33 +178,41 @@ const (
|
||||||
out.write('\tIRQ_max = {} // Highest interrupt number on this device.\n'.format(intrMax))
|
out.write('\tIRQ_max = {} // Highest interrupt number on this device.\n'.format(intrMax))
|
||||||
out.write(')\n')
|
out.write(')\n')
|
||||||
|
|
||||||
out.write('\n// Peripherals\nvar (')
|
|
||||||
for peripheral in device.peripherals:
|
for peripheral in device.peripherals:
|
||||||
out.write('\n\t// {description}\n\t{name} = struct {{\n'.format(**peripheral))
|
out.write('\n// {description}\ntype {name}_Type struct {{\n'.format(**peripheral))
|
||||||
|
address = peripheral['baseAddress']
|
||||||
|
padNumber = 0
|
||||||
for register in peripheral['registers']:
|
for register in peripheral['registers']:
|
||||||
for variant in register['variants']:
|
if address > register['address']:
|
||||||
regType = '__reg'
|
# In Nordic SVD files, these registers are deprecated or
|
||||||
if register['array'] is not None:
|
# duplicates, so can be ignored.
|
||||||
regType = '[{}]__reg'.format(register['array'])
|
#print('skip: %s.%s' % (peripheral['name'], register['name']))
|
||||||
out.write('\t\t{name} {regType}\n'.format(**variant, regType=regType))
|
continue
|
||||||
out.write('\t}{\n')
|
|
||||||
for register in peripheral['registers']:
|
# insert padding, if needed
|
||||||
for variant in register['variants']:
|
if address < register['address']:
|
||||||
out.write('\t\t{name}: '.format(**variant))
|
numSkip = (register['address'] - address) // 4
|
||||||
if register['array'] is not None:
|
if numSkip == 1:
|
||||||
out.write('[{num}]__reg{{'.format(num=register['array']))
|
out.write('\t_padding{padNumber} __reg\n'.format(padNumber=padNumber))
|
||||||
if register['description']:
|
|
||||||
out.write(' // {description}'.format(**register))
|
|
||||||
out.write('\n')
|
|
||||||
for i in range(register['array']):
|
|
||||||
out.write('\t\t\t0x{:x},\n'.format(variant['address'] + i * 4)) # TODO: pointer width
|
|
||||||
out.write('\t\t},')
|
|
||||||
else:
|
else:
|
||||||
out.write('0x{address:x},'.format(**variant))
|
out.write('\t_padding{padNumber} [{num}]__reg\n'.format(padNumber=padNumber, num=numSkip))
|
||||||
if register['description']:
|
padNumber += 1
|
||||||
out.write(' // {description}'.format(**register))
|
|
||||||
out.write('\n')
|
regType = '__reg'
|
||||||
out.write('\t}\n')
|
if register['array'] is not None:
|
||||||
|
regType = '[{}]__reg'.format(register['array'])
|
||||||
|
out.write('\t{name} {regType}\n'.format(**register, regType=regType))
|
||||||
|
|
||||||
|
# next address
|
||||||
|
if register['array'] is not None and 1:
|
||||||
|
address = register['address'] + 4 * register['array']
|
||||||
|
else:
|
||||||
|
address = register['address'] + 4
|
||||||
|
out.write('}\n')
|
||||||
|
|
||||||
|
out.write('\n// Peripherals.\nvar (\n')
|
||||||
|
for peripheral in device.peripherals:
|
||||||
|
out.write('\t{name} = (*{name}_Type)(unsafe.Pointer(uintptr(0x{baseAddress:x}))) // {description}\n'.format(**peripheral))
|
||||||
out.write(')\n')
|
out.write(')\n')
|
||||||
|
|
||||||
for peripheral in device.peripherals:
|
for peripheral in device.peripherals:
|
||||||
|
@ -316,44 +220,24 @@ const (
|
||||||
out.write('\n// Bitfields for {name}: {description}\nconst('.format(**peripheral))
|
out.write('\n// Bitfields for {name}: {description}\nconst('.format(**peripheral))
|
||||||
for register in peripheral['registers']:
|
for register in peripheral['registers']:
|
||||||
if not register['bitfields']: continue
|
if not register['bitfields']: continue
|
||||||
for variant in register['variants']:
|
out.write('\n\t// {name}'.format(**register))
|
||||||
out.write('\n\t// {name}'.format(**variant))
|
if register['description']:
|
||||||
if register['description']:
|
out.write(': {description}'.format(**register))
|
||||||
out.write(': {description}'.format(**register))
|
out.write('\n')
|
||||||
out.write('\n')
|
|
||||||
for bitfield in register['bitfields']:
|
for bitfield in register['bitfields']:
|
||||||
out.write('\t{name} = 0x{value:x}'.format(**bitfield))
|
out.write('\t{name} = 0x{value:x}'.format(**bitfield))
|
||||||
if bitfield['description']:
|
if bitfield['description']:
|
||||||
out.write('// {description}'.format(**bitfield))
|
out.write(' // {description}'.format(**bitfield))
|
||||||
out.write('\n')
|
out.write('\n')
|
||||||
out.write(')\n')
|
out.write(')\n')
|
||||||
|
|
||||||
def writeLD(outdir, device):
|
|
||||||
# Variables for the linker script.
|
|
||||||
out = open(outdir + '/' + device.metadata['nameLower'] + '.ld', 'w')
|
|
||||||
out.write('''\
|
|
||||||
/* Automatically generated file. DO NOT EDIT. */
|
|
||||||
/* Generated by gen-device.py from {file}, see {descriptorSource} */
|
|
||||||
|
|
||||||
__flash_size = 0x{flashSize:x};
|
|
||||||
__ram_size = 0x{ramSize:x};
|
|
||||||
__num_isrs = {numInterrupts};
|
|
||||||
'''.format(**device.metadata))
|
|
||||||
out.close()
|
|
||||||
|
|
||||||
|
|
||||||
def generate(indir, outdir):
|
def generate(indir, outdir):
|
||||||
for filepath in glob(indir + '/*.svd'):
|
for filepath in sorted(glob(indir + '/*.svd')):
|
||||||
print(filepath)
|
print(filepath)
|
||||||
device = readSVD(filepath)
|
device = readSVD(filepath)
|
||||||
writeGo(outdir, device)
|
writeGo(outdir, device)
|
||||||
|
|
||||||
for filepath in glob(indir + '/*.atdf'):
|
|
||||||
print(filepath)
|
|
||||||
device = readATDF(filepath)
|
|
||||||
writeGo(outdir, device)
|
|
||||||
writeLD(outdir, device)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
indir = sys.argv[1] # directory with register descriptor files (*.svd, *.atdf)
|
indir = sys.argv[1] # directory with register descriptor files (*.svd, *.atdf)
|
|
@ -29,6 +29,10 @@
|
||||||
// POSSIBILITY OF SUCH DAMAGE.
|
// POSSIBILITY OF SUCH DAMAGE.
|
||||||
package arm
|
package arm
|
||||||
|
|
||||||
|
import (
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
type __reg uint32
|
type __reg uint32
|
||||||
type RegValue = __reg
|
type RegValue = __reg
|
||||||
|
|
||||||
|
@ -42,22 +46,13 @@ const (
|
||||||
)
|
)
|
||||||
|
|
||||||
// Nested Vectored Interrupt Controller (NVIC).
|
// Nested Vectored Interrupt Controller (NVIC).
|
||||||
var NVIC = struct {
|
type NVIC_Type struct {
|
||||||
ISER [8]__reg
|
ISER [8]__reg
|
||||||
}{
|
|
||||||
ISER: [8]__reg{
|
|
||||||
NVIC_BASE + 0x000,
|
|
||||||
NVIC_BASE + 0x004,
|
|
||||||
NVIC_BASE + 0x008,
|
|
||||||
NVIC_BASE + 0x00C,
|
|
||||||
NVIC_BASE + 0x010,
|
|
||||||
NVIC_BASE + 0x014,
|
|
||||||
NVIC_BASE + 0x018,
|
|
||||||
NVIC_BASE + 0x01C,
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var NVIC = (*NVIC_Type)(unsafe.Pointer(uintptr(NVIC_BASE)))
|
||||||
|
|
||||||
// Enable the given interrupt number.
|
// Enable the given interrupt number.
|
||||||
func EnableIRQ(irq uint32) {
|
func EnableIRQ(irq uint32) {
|
||||||
NVIC.ISER[irq >> 5] = 1 << (irq & 0x1F)
|
NVIC.ISER[irq>>5] = 1 << (irq & 0x1F)
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,15 +19,15 @@ const LED = 13
|
||||||
func (p GPIO) Configure(config GPIOConfig) {
|
func (p GPIO) Configure(config GPIOConfig) {
|
||||||
if config.Mode == GPIO_OUTPUT { // set output bit
|
if config.Mode == GPIO_OUTPUT { // set output bit
|
||||||
if p.Pin < 8 {
|
if p.Pin < 8 {
|
||||||
avr.PORT.DDRD |= 1 << p.Pin
|
*avr.DDRD |= 1 << p.Pin
|
||||||
} else {
|
} else {
|
||||||
avr.PORT.DDRB |= 1 << (p.Pin - 8)
|
*avr.DDRB |= 1 << (p.Pin - 8)
|
||||||
}
|
}
|
||||||
} else { // configure input: clear output bit
|
} else { // configure input: clear output bit
|
||||||
if p.Pin < 8 {
|
if p.Pin < 8 {
|
||||||
avr.PORT.DDRD &^= 1 << p.Pin
|
*avr.DDRD &^= 1 << p.Pin
|
||||||
} else {
|
} else {
|
||||||
avr.PORT.DDRB &^= 1 << (p.Pin - 8)
|
*avr.DDRB &^= 1 << (p.Pin - 8)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -35,15 +35,15 @@ func (p GPIO) Configure(config GPIOConfig) {
|
||||||
func (p GPIO) Set(value bool) {
|
func (p GPIO) Set(value bool) {
|
||||||
if value { // set bits
|
if value { // set bits
|
||||||
if p.Pin < 8 {
|
if p.Pin < 8 {
|
||||||
avr.PORT.PORTD |= 1 << p.Pin
|
*avr.PORTD |= 1 << p.Pin
|
||||||
} else {
|
} else {
|
||||||
avr.PORT.PORTB |= 1 << (p.Pin - 8)
|
*avr.PORTB |= 1 << (p.Pin - 8)
|
||||||
}
|
}
|
||||||
} else { // clear bits
|
} else { // clear bits
|
||||||
if p.Pin < 8 {
|
if p.Pin < 8 {
|
||||||
avr.PORT.PORTB &^= 1 << p.Pin
|
*avr.PORTB &^= 1 << p.Pin
|
||||||
} else {
|
} else {
|
||||||
avr.PORT.PORTB &^= 1 << (p.Pin - 8)
|
*avr.PORTB &^= 1 << (p.Pin - 8)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -36,17 +36,17 @@ func init() {
|
||||||
|
|
||||||
func initUART() {
|
func initUART() {
|
||||||
// Initialize UART at 115200 baud when running at 16MHz.
|
// Initialize UART at 115200 baud when running at 16MHz.
|
||||||
avr.USART.UBRR0H = 0
|
*avr.UBRR0H = 0
|
||||||
avr.USART.UBRR0L = 8
|
*avr.UBRR0L = 8
|
||||||
avr.USART.UCSR0B = avr.UCSR0B_RXEN0 | avr.UCSR0B_TXEN0 // enable RX and TX
|
*avr.UCSR0B = avr.UCSR0B_RXEN0 | avr.UCSR0B_TXEN0 // enable RX and TX
|
||||||
avr.USART.UCSR0C = avr.UCSR0C_UCSZ0 // 8-bits data
|
*avr.UCSR0C = avr.UCSR0C_UCSZ0 // 8-bits data
|
||||||
}
|
}
|
||||||
|
|
||||||
func putchar(c byte) {
|
func putchar(c byte) {
|
||||||
for (avr.USART.UCSR0A & avr.UCSR0A_UDRE0) == 0 {
|
for (*avr.UCSR0A & avr.UCSR0A_UDRE0) == 0 {
|
||||||
// Wait until previous char has been sent.
|
// Wait until previous char has been sent.
|
||||||
}
|
}
|
||||||
avr.USART.UDR0 = avr.RegValue(c) // send char
|
*avr.UDR0 = avr.RegValue(c) // send char
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sleep by the given amount.
|
// Sleep by the given amount.
|
||||||
|
@ -73,21 +73,21 @@ func sleepWDT(period uint8) {
|
||||||
avr.Asm("cli")
|
avr.Asm("cli")
|
||||||
avr.Asm("wdr")
|
avr.Asm("wdr")
|
||||||
// Start timed sequence.
|
// Start timed sequence.
|
||||||
avr.WDT.WDTCSR |= avr.WDTCSR_WDCE | avr.WDTCSR_WDE
|
*avr.WDTCSR |= avr.WDTCSR_WDCE | avr.WDTCSR_WDE
|
||||||
// Enable WDT and set new timeout (0.5s)
|
// Enable WDT and set new timeout (0.5s)
|
||||||
avr.WDT.WDTCSR = avr.WDTCSR_WDIE | avr.RegValue(period)
|
*avr.WDTCSR = avr.WDTCSR_WDIE | avr.RegValue(period)
|
||||||
avr.Asm("sei")
|
avr.Asm("sei")
|
||||||
|
|
||||||
// Set sleep mode to idle and enable sleep mode.
|
// Set sleep mode to idle and enable sleep mode.
|
||||||
// Note: when using something other than idle, the UART won't work
|
// Note: when using something other than idle, the UART won't work
|
||||||
// correctly. This needs to be fixed, though, so we can truly sleep.
|
// correctly. This needs to be fixed, though, so we can truly sleep.
|
||||||
avr.CPU.SMCR = (0 << 1) | avr.SMCR_SE
|
*avr.SMCR = (0 << 1) | avr.SMCR_SE
|
||||||
|
|
||||||
// go to sleep
|
// go to sleep
|
||||||
avr.Asm("sleep")
|
avr.Asm("sleep")
|
||||||
|
|
||||||
// disable sleep
|
// disable sleep
|
||||||
avr.CPU.SMCR = 0
|
*avr.SMCR = 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func monotime() uint64 {
|
func monotime() uint64 {
|
||||||
|
|
Загрузка…
Создание таблицы
Сослаться в новой задаче