-
Notifications
You must be signed in to change notification settings - Fork 36
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Alternative SVD2Ada output for component drivers #40
Comments
On 15 Jun 2017, at 13:47, Fabien Chouteau ***@***.***> wrote:
This week-end I was writing a driver for what we call a component in the
Ada_Drivers_Library. It's an external chip that communicates with the CPU
though communication buses (I2C, SPI, UART).
Most of the time, writing a driver for these components is very similar to
writing a driver for the internal devices of the MCU. There's a bunch of
registers of which you have to change some of the fields to a given value.
The big difference is that those registers are not mapped in the CPU address
space, so it means we cannot directly use the representation clauses like we
do for the internal devices.
What we do now is:
Read the raw value of the register
Use bit shifts and masks like we would do in C...
Write back the register
So I was writing another driver and I was thinking that it is actually possible
to use representation clauses and do unchecked conversion to and from the raw
representation. It's just very tedious to write it manually.
Personally I've done just this; not _that_ painful! But certainly a pain that could be eased. I'm not sure that writing XML is better than writing Ada, though, so you'd have to hope for re-use.
The idea is to use SVD2Ada to produce a slightly different output that can be
used to write better component drivers.
SVD2Ada would generate:
Representation of the registers (that's already implemented)
Sub programs to read and write those registers using Unchecked_Conversion
I suppose it'd depend on whether it's easier to include a mode switch in svd2ada or to extract the relevant parts into a library, then use the library to support a separate tool.
with HAL; use HAL;
with System; use System;
[[[I know you guys use 'use' a lot (esp. in the compiler), but it really makes it hard to find stuff. (e.g. why not HAL.UInt8 likeHAL.UInt2 below)]]]
generic
type Driver (<>) is limited private;
with function Read_Register (This : in out Driver; Addr : UInt8) return UInt8;
with procedure Write_Register (This : in out Driver; Addr : UInt8; Data : UInt8);
package SGTL5000_Registers is
SR_Register_Address : constant := 16#0F#;
Why in the spec?
type SR_Register is record
AWD : Boolean := False;
EOC : Boolean := False;
JEOC : Boolean := False;
JSTRT : Boolean := False;
STRT : Boolean := False;
OVR : Boolean := False;
Reserved_6_7 : HAL.UInt2 := 16#0#;
end record
with Volatile_Full_Access, Size => 8,
Bit_Order => System.Low_Order_First;
Doesn't need to be volatile, nowhere near the hardware. Of course you need to do an explicit read-modify-write for a bitfield access in an external register.
Sounds like a useful idea!
|
Yes, I do this as well and then you can validate e.g. sensor registers received via network packets or I2C bytes to some degree. With 'Valid_Scalars Though I am not sure if I prefer having the conversions and records local to the sensor package in these cases as they are hand constructed from reference manuals. I guess the proof is in the pudding. I would certainly prefer writing Ada records over XML. Especially as getting the sizes right on a large register, can be tedious sometimes (but it is fantastic, when it finally compiles).
Agreed, Especially on github or if the language server is failing to look things up due to code breakage. Then it can be really annoying. Isn't it optimising for the writer? I also had a weird issue with svd2ada yesterday, where I would have to build from the terminal as for whatever reason gnat studio compilation would seemingly randomly decide that some of it's codebase was ambiguous (vector .append). |
This week-end I was writing a driver for what we call a component in the
Ada_Drivers_Library. It's an external chip that communicates with the CPU
though communication buses (I2C, SPI, UART).
Most of the time, writing a driver for these components is very similar to
writing a driver for the internal devices of the MCU. There's a bunch of
registers of which you have to change some of the fields to a given value.
The big difference is that those registers are not mapped in the CPU address
space, so it means we cannot directly use the representation clauses like we
do for the internal devices.
What we do now is:
So I was writing another driver and I was thinking that it is actually possible
to use representation clauses and do unchecked conversion to and from the raw
representation. It's just very tedious to write it manually.
The idea is to use SVD2Ada to produce a slightly different output that can be
used to write better component drivers.
SVD2Ada would generate:
I attach and a prototype of what it could look like. There are 4 files:
sgtl5000_registers.ads: generated by SVD2Ada. It contains definition of the
registers and the sub-programs to read and write them. It's a generic package
so the developer of the driver can define the sub-programs to read and write
raw register data.
sgtl5000_registers.adb: generated by SVD2Ada. It contains the implementation
of the sub-programs to read and write registers. To do that it uses
Ada.Unchecked_Conversion and the sub-programs provided by the user
sgtl5000.ads: written by the developer. The interface of the driver.
sgtl5000.adb: written by the developer. It uses SGTL5000_Register to
implement the driver.
The text was updated successfully, but these errors were encountered: