Skip to content
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

extract enums (do not merge) #235

Draft
wants to merge 2 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
27 changes: 27 additions & 0 deletions src/cli.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
use anyhow::{Ok, Result};
use clap::Parser;
use std::{fs::File, io::Write, path::PathBuf, str::FromStr};
use svd_rs::ValidateLevel;

use svdtools::{
convert::convert_cli,
enum_extract::enum_extract,
html::html_cli,
html::htmlcompare_cli,
info,
Expand Down Expand Up @@ -134,6 +136,16 @@ enum Command {
/// Describe requested information
request: String,
},
/// Makes patch file with enums extracted from SVD
ExtractEnums {
/// Path to input file
in_path: PathBuf,
/// Format of input file (XML, JSON or YAML)
#[clap(long = "input-format")]
input_format: Option<convert_cli::InputFormat>,
/// Path to output file
out_path: PathBuf,
},
}

impl Command {
Expand Down Expand Up @@ -200,6 +212,7 @@ impl Command {
expand: *expand,
expand_properties: *expand_properties,
ignore_enums: *ignore_enums,
validate_level: ValidateLevel::Disabled,
},
format_config.as_ref().map(|p| p.as_path()),
)?,
Expand All @@ -226,6 +239,20 @@ impl Command {
let response = request.process(&device)?;
print!("{response}")
}
Self::ExtractEnums {
in_path,
input_format,
out_path,
} => {
let mut cfg = convert_cli::ParserConfig::default();
cfg.validate_level = ValidateLevel::Disabled;
let device = convert_cli::open_svd(in_path, *input_format, cfg)?;
let yml = enum_extract(&device);
let mut out_str = String::new();
let mut emitter = yaml_rust::YamlEmitter::new(&mut out_str);
emitter.dump(&yml).unwrap();
File::create(out_path)?.write_all(out_str.as_bytes())?;
}
}
Ok(())
}
Expand Down
7 changes: 5 additions & 2 deletions src/convert/convert_cli.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ use anyhow::{anyhow, Result};
use std::io::{Read, Write};
use std::str::FromStr;
use std::{fs::File, path::Path};
use svd_rs::Device;
use svd_rs::{Device, ValidateLevel};

use crate::get_encoder_config;
pub use crate::ConfigFormat;
Expand Down Expand Up @@ -52,6 +52,7 @@ pub struct ParserConfig {
pub expand: bool,
pub expand_properties: bool,
pub ignore_enums: bool,
pub validate_level: ValidateLevel,
}

pub fn open_svd(
Expand All @@ -73,7 +74,9 @@ pub fn open_svd(
let mut device = match input_format {
InputFormat::Xml => svd_parser::parse_with_config(
&input,
&svd_parser::Config::default().ignore_enums(parser_config.ignore_enums),
&svd_parser::Config::default()
.ignore_enums(parser_config.ignore_enums)
.validate_level(parser_config.validate_level),
)?,
InputFormat::Yaml => serde_yaml::from_str(&input)?,
InputFormat::Json => serde_json::from_str(&input)?,
Expand Down
144 changes: 144 additions & 0 deletions src/enum_extract/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,144 @@
use crate::patch::ToYaml;
use svd_rs::EnumeratedValues;
use yaml_rust::yaml::{self, Yaml};

pub trait HasEnums {
fn has_enums(&self) -> bool;
}

impl HasEnums for svd_rs::Field {
fn has_enums(&self) -> bool {
!self.enumerated_values.is_empty()
}
}

impl HasEnums for svd_rs::Register {
fn has_enums(&self) -> bool {
if let Some(fields) = self.fields.as_ref() {
for f in fields {
if f.has_enums() {
return true;
}
}
}
false
}
}

impl HasEnums for svd_rs::RegisterCluster {
fn has_enums(&self) -> bool {
match self {
svd_rs::RegisterCluster::Cluster(c) => c.has_enums(),
svd_rs::RegisterCluster::Register(r) => r.has_enums(),
}
}
}

impl HasEnums for svd_rs::Cluster {
fn has_enums(&self) -> bool {
for rc in &self.children {
if rc.has_enums() {
return true;
}
}
false
}
}

impl HasEnums for svd_rs::Peripheral {
fn has_enums(&self) -> bool {
if let Some(regs) = self.registers.as_ref() {
for rc in regs {
if rc.has_enums() {
return true;
}
}
}
false
}
}

fn evs_to_hash(evs: &EnumeratedValues) -> yaml::Hash {
let mut hash = yaml::Hash::with_capacity(evs.values.len());
if let Some(n) = evs.name.as_ref() {
hash.insert("_name".to_yaml(), n.to_yaml());
}
if let Some(d) = evs.derived_from.as_ref() {
hash.insert("_derivedFrom".to_yaml(), d.to_yaml());
} else {
for ev in &evs.values {
let val = if let Some(val) = ev.value {
Yaml::Integer(val as _)
} else if ev.is_default() {
Yaml::Integer(-1)
} else {
panic!("EnumeratedValue without value");
};
hash.insert(
ev.name.to_yaml(),
Yaml::Array(vec![val, ev.description.as_deref().unwrap_or("").to_yaml()]),
);
}
}
hash
}

fn rc_enum_extact(regs: &[svd_rs::RegisterCluster]) -> Yaml {
let mut phash = yaml::Hash::new();
let mut pchash = yaml::Hash::new();
for rc in regs {
if rc.has_enums() {
match rc {
svd_rs::RegisterCluster::Cluster(c) => {
pchash.insert(c.name.to_yaml(), rc_enum_extact(&c.children));
}
svd_rs::RegisterCluster::Register(r) => {
let mut rhash = yaml::Hash::new();
for f in r.fields() {
if f.has_enums() {
let mut fhash = yaml::Hash::with_capacity(f.enumerated_values.len());
for evs in &f.enumerated_values {
match evs.usage {
Some(svd_rs::Usage::Read) => {
fhash.insert(
"_read".to_yaml(),
Yaml::Hash(evs_to_hash(evs)),
);
}
Some(svd_rs::Usage::Write) => {
fhash.insert(
"_write".to_yaml(),
Yaml::Hash(evs_to_hash(evs)),
);
}
_ => {
assert_eq!(f.enumerated_values.len(), 1);
fhash.extend(evs_to_hash(evs));
}
}
}
rhash.insert(f.name.to_yaml(), Yaml::Hash(fhash));
}
}
phash.insert(r.name.to_yaml(), Yaml::Hash(rhash));
}
}
}
}
if !pchash.is_empty() {
phash.insert("_clusters".to_yaml(), Yaml::Hash(pchash));
}
Yaml::Hash(phash)
}

pub fn enum_extract(device: &svd_rs::Device) -> Yaml {
let mut hash = yaml::Hash::new();
for p in &device.peripherals {
if let Some(regs) = p.registers.as_ref() {
if p.has_enums() {
hash.insert(p.name.to_yaml(), rc_enum_extact(regs));
}
}
}
Yaml::Hash(hash)
}
1 change: 1 addition & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ use std::{fs::File, path::Path, str::FromStr};

pub mod common;
pub mod convert;
pub mod enum_extract;
pub mod html;
pub mod info;
pub mod interrupts;
Expand Down
3 changes: 2 additions & 1 deletion src/patch/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,8 @@ mod iterators;
mod peripheral;
mod register;
mod yaml_ext;
use yaml_ext::{AsType, GetVal, ToYaml};
pub(crate) use yaml_ext::ToYaml;
use yaml_ext::{AsType, GetVal};

use crate::get_encoder_config;

Expand Down
Loading