opentitanlib/transport/hyperdebug/
servo_micro.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
// Copyright lowRISC contributors (OpenTitan project).
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0

use anyhow::{bail, Result};
use std::rc::Rc;

use crate::io::gpio::{GpioPin, PinMode, PullMode};
use crate::transport::hyperdebug::{Flavor, Inner, StandardFlavor, VID_GOOGLE};
use crate::transport::{TransportError, TransportInterfaceType};

/// The Servo Micro is used to bring up GSC and EC chips sitting inside a computing device, such
/// that those GSC chips can provide Case Closed Debugging support to allow bringup of the rest of
/// the computing device. Servo devices happen to speak almost exactly the same USB protocol as
/// HyperDebug. This `Flavor` implementation defines the few deviations: USB PID value, and the
/// handling of OT reset signal.
pub struct ServoMicroFlavor;

impl ServoMicroFlavor {
    const PID_SERVO_MICRO: u16 = 0x501A;
}

impl Flavor for ServoMicroFlavor {
    fn gpio_pin(inner: &Rc<Inner>, pinname: &str) -> Result<Rc<dyn GpioPin>> {
        if pinname == "IO_EXP_16" {
            return Ok(Rc::new(ServoMicroResetPin::open(inner)?));
        }
        StandardFlavor::gpio_pin(inner, pinname)
    }

    fn spi_index(_inner: &Rc<Inner>, instance: &str) -> Result<(u8, u8)> {
        bail!(TransportError::InvalidInstance(
            TransportInterfaceType::Spi,
            instance.to_string()
        ))
    }

    fn get_default_usb_vid() -> u16 {
        VID_GOOGLE
    }

    fn get_default_usb_pid() -> u16 {
        Self::PID_SERVO_MICRO
    }
    fn perform_initial_fw_check() -> bool {
        // The servo micro firmware and hyperdebug firmware are different, so do not check.
        false
    }
}

/// A register for the IO expander on the servo micro i2c bus
#[derive(Copy, Clone)]
#[repr(u8)]
enum IoExpanderRegister {
    /// Input values for Port 1 pins. `1'b` represents a logic high.
    Input1 = 1,
    /// Output values for Port 1 pins. `1'b` represents a logic high. Note that if `FloatPin1` bit
    /// is set to high, then output pin value is not used.
    Output1 = 3,
    /// Specifies if the IO expander is floating or driving the pin. `1'b` represents the IO
    /// expander producing a HighZ floating state on the pin, and `0'b` represents the IO expander
    /// driving the value from the `Output1` register on the pin.
    FloatPin1 = 7,
}

/// Handles the specialized IO expander logic to read and write to the OT reset pin
pub struct ServoMicroResetPin {
    inner: Rc<Inner>,
}

impl ServoMicroResetPin {
    /// The target reset pin mask for `IoExpanderRegister `
    const RESET_PIN_MASK: u8 = 1 << 6;

    pub fn open(inner: &Rc<Inner>) -> Result<Self> {
        Ok(Self {
            inner: Rc::clone(inner),
        })
    }

    /// Returns the raw value of `IoExpanderRegister`.
    fn read_reg(&self, reg: IoExpanderRegister) -> Result<u8> {
        let cmd = format!("i2cxfer r 0 0x20 {}", reg as u8);
        let line = self
            .inner
            .cmd_one_line_output(&cmd)
            .map_err(|_| TransportError::CommunicationError(format!("No output from {cmd}")))?;
        let Some(Ok(val)) = line
            .split_ascii_whitespace()
            .last()
            .map(|s| s.trim_matches(|c| !char::is_ascii_digit(&c)).parse::<u8>())
        else {
            bail!(TransportError::CommunicationError(format!(
                "Bad i2cxfer output: '{line}'"
            )))
        };
        Ok(val)
    }

    /// Writes the entire raw value to a `IoExpanderRegister`.
    fn write_reg(&self, reg: IoExpanderRegister, val: u8) -> Result<()> {
        let cmd = format!("i2cxfer w 0 0x20 {} {val}", reg as u8);
        self.inner.cmd_no_output(&cmd)
    }

    /// Reads only the value of OT reset pin for the specified `IoExpanderRegister`.
    fn read_pin(&self, reg: IoExpanderRegister) -> Result<bool> {
        self.read_reg(reg)
            .map(|val| val & Self::RESET_PIN_MASK != 0)
    }

    /// Writes only the value of the OT reset pin for the specified `IoExpanderRegister` via
    /// read-modify-write operation on the `IoExpanderRegister`.
    fn write_pin(&self, reg: IoExpanderRegister, val: bool) -> Result<()> {
        let read_val = self.read_reg(reg)?;
        let write_val = if val {
            read_val | Self::RESET_PIN_MASK
        } else {
            read_val & !Self::RESET_PIN_MASK
        };
        // Only write if the value isn't already what we want
        if read_val != write_val {
            self.write_reg(reg, write_val)
        } else {
            Ok(())
        }
    }
}

impl GpioPin for ServoMicroResetPin {
    /// Reads the value of the reset pin.
    fn read(&self) -> Result<bool> {
        self.read_pin(IoExpanderRegister::Input1)
    }

    /// Sets the value of the GPIO reset pin by means of multiple io expander commands
    fn write(&self, value: bool) -> Result<()> {
        // We don't need to write to the output register if we are just going to float the pin
        if !value {
            self.write_pin(IoExpanderRegister::Output1, false)?;
        }
        self.write_pin(IoExpanderRegister::FloatPin1, value)
    }

    fn set_mode(&self, _mode: PinMode) -> Result<()> {
        bail!(TransportError::UnsupportedOperation)
    }

    fn set_pull_mode(&self, _mode: PullMode) -> Result<()> {
        bail!(TransportError::UnsupportedOperation)
    }
}