cube-query-lib 0.1.1

A tool for querying the CubeMX database
Documentation
pub use clap;
use clap::Parser;
use prettytable::format::consts::FORMAT_CLEAN;
use prettytable::{row, Attr, Cell, Row, Table};

mod parse_ip;
mod parse_mcu;
mod paths;

#[derive(Debug, Parser)]
pub struct Args {
    pub chip: String,
    pub filter: Option<String>,
}

struct PinRow {
    pin: String,
    r#use: String,
    mode: String,
}

pub fn run(args: &Args) -> anyhow::Result<()> {
    let filter = args.filter.clone().unwrap_or_default().to_ascii_lowercase();
    let db = paths::obtain_db(&args.chip)?;
    let db = std::fs::read_to_string(db)?;

    let db = std::thread::spawn(move || parse_mcu::parse_xml(&db));

    let gpios = std::thread::spawn(move || parse_ip::parse_ip());

    let db = db.join().unwrap()?;
    let gpios = gpios.join().unwrap()?;

    // convert gpios into HashMap name: Vec<PinSignal>
    let gpios = gpios
        .gpio_pins
        .into_iter()
        .map(|pin| (pin.name.to_string(), pin.pin_signals))
        .collect::<std::collections::HashMap<_, _>>();

    let mut pin_row = Vec::with_capacity(db.pins.len());

    for pin in db.pins {
        for signal in pin.signals {
            let signal_name = signal.name.to_ascii_lowercase();
            if signal_name.contains(&filter) {
                let gpio = gpios.get(&pin.name).unwrap();

                let signal = gpio
                    .iter()
                    .find(|signal| signal.name.to_ascii_lowercase() == signal_name)
                    .unwrap();

                let signal_value = signal
                    .specific_parameters
                    .iter()
                    .flat_map(|p| &p.possible_values)
                    .next()
                    .unwrap();

                let (_, signal_value, _) = signal_name_components(signal_value).unwrap();

                pin_row.push(PinRow {
                    pin: pin.name.to_string(),
                    r#use: signal.name.to_string(),
                    mode: signal_value.to_string(),
                });
            }
        }
    }

    pin_row.sort_by_key(|row| row.r#use.to_ascii_lowercase());

    let mut table = Table::new();
    table.set_format(*FORMAT_CLEAN);
    table.add_row(Row::new(vec![
        Cell::new("Use").with_style(Attr::Bold),
        Cell::new("Pin").with_style(Attr::Bold),
        Cell::new("Mode").with_style(Attr::Bold),
    ]));
    for row in pin_row {
        table.add_row(row![row.r#use, row.pin, row.mode]);
    }

    table.printstd();

    Ok(())
}

/// name: GPIO_{X}_{Y}
fn signal_name_components(name: &str) -> Option<(&str, &str, &str)> {
    let mut parts = name.split('_');
    let gpio = parts.next()?;
    let x = parts.next()?;
    let y = parts.next()?;
    Some((gpio, x, y))
}