opregion.rs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. use crate::{
  2. value::{Args, FieldAccessType, FieldFlags, FieldUpdateRule},
  3. AmlContext,
  4. AmlError,
  5. AmlName,
  6. AmlValue,
  7. };
  8. use bit_field::BitField;
  9. #[derive(Clone, Debug)]
  10. pub struct OpRegion {
  11. region: RegionSpace,
  12. base: u64,
  13. length: u64,
  14. parent_device: Option<AmlName>,
  15. }
  16. impl OpRegion {
  17. pub fn new(region: RegionSpace, base: u64, length: u64, parent_device: Option<AmlName>) -> OpRegion {
  18. OpRegion { region, base, length, parent_device }
  19. }
  20. /// Get the length of this op-region, in **bits**.
  21. pub fn length(&self) -> u64 {
  22. self.length
  23. }
  24. /// Read a field from this op-region. This has looser requirements than `read`, and will
  25. /// perform multiple standard-sized reads and mask the result as required.
  26. pub fn read_field(
  27. &self,
  28. offset: u64,
  29. length: u64,
  30. flags: FieldFlags,
  31. context: &mut AmlContext,
  32. ) -> Result<AmlValue, AmlError> {
  33. let _max_access_size = match self.region {
  34. RegionSpace::SystemMemory => 64,
  35. RegionSpace::SystemIo | RegionSpace::PciConfig => 32,
  36. _ => unimplemented!(),
  37. };
  38. let minimum_access_size = match flags.access_type()? {
  39. FieldAccessType::Any => 8,
  40. FieldAccessType::Byte => 8,
  41. FieldAccessType::Word => 16,
  42. FieldAccessType::DWord => 32,
  43. FieldAccessType::QWord => 64,
  44. FieldAccessType::Buffer => 8, // TODO
  45. };
  46. /*
  47. * Find the access size, as either the minimum access size allowed by the region, or the field length
  48. * rounded up to the next power-of-2, whichever is larger.
  49. */
  50. let access_size = u64::max(minimum_access_size, length.next_power_of_two());
  51. /*
  52. * TODO: we need to decide properly how to read from the region itself. Complications:
  53. * - if the region has a minimum access size greater than the desired length, we need to read the
  54. * minimum and mask it (reading a byte from a WordAcc region)
  55. * - if the desired length is larger than we can read, we need to do multiple reads
  56. */
  57. let value = self.read(offset, access_size, context)?.get_bits(0..(length as usize));
  58. Ok(AmlValue::Integer(value))
  59. }
  60. pub fn write_field(
  61. &self,
  62. offset: u64,
  63. length: u64,
  64. flags: FieldFlags,
  65. value: AmlValue,
  66. context: &mut AmlContext,
  67. ) -> Result<(), AmlError> {
  68. /*
  69. * If the field's update rule is `Preserve`, we need to read the initial value of the field, so we can
  70. * overwrite the correct bits. We destructure the field to do the actual write, so we read from it if
  71. * needed here, otherwise the borrow-checker doesn't understand.
  72. */
  73. let mut field_value = match flags.field_update_rule()? {
  74. FieldUpdateRule::Preserve => self.read_field(offset, length, flags, context)?.as_integer(context)?,
  75. FieldUpdateRule::WriteAsOnes => 0xffffffff_ffffffff,
  76. FieldUpdateRule::WriteAsZeros => 0x0,
  77. };
  78. let _maximum_access_size = match self.region {
  79. RegionSpace::SystemMemory => 64,
  80. RegionSpace::SystemIo | RegionSpace::PciConfig => 32,
  81. _ => unimplemented!(),
  82. };
  83. let minimum_access_size = match flags.access_type()? {
  84. FieldAccessType::Any => 8,
  85. FieldAccessType::Byte => 8,
  86. FieldAccessType::Word => 16,
  87. FieldAccessType::DWord => 32,
  88. FieldAccessType::QWord => 64,
  89. FieldAccessType::Buffer => 8, // TODO
  90. };
  91. /*
  92. * Find the access size, as either the minimum access size allowed by the region, or the field length
  93. * rounded up to the next power-of-2, whichever is larger.
  94. */
  95. let access_size = u64::max(minimum_access_size, length.next_power_of_two());
  96. field_value.set_bits(0..(length as usize), value.as_integer(context)?);
  97. self.write(offset, access_size, field_value, context)
  98. }
  99. /// Perform a standard-size read from this op-region. `length` must be a supported power-of-2,
  100. /// and `offset` correctly aligned for that `length`. `value` must be appropriately sized.
  101. pub fn read(&self, offset: u64, length: u64, context: &mut AmlContext) -> Result<u64, AmlError> {
  102. match self.region {
  103. RegionSpace::SystemMemory => {
  104. let address = (self.base + offset).try_into().map_err(|_| AmlError::FieldInvalidAddress)?;
  105. match length {
  106. 8 => Ok(context.handler.read_u8(address) as u64),
  107. 16 => Ok(context.handler.read_u16(address) as u64),
  108. 32 => Ok(context.handler.read_u32(address) as u64),
  109. 64 => Ok(context.handler.read_u64(address)),
  110. _ => Err(AmlError::FieldInvalidAccessSize),
  111. }
  112. }
  113. RegionSpace::SystemIo => {
  114. let port = (self.base + offset).try_into().map_err(|_| AmlError::FieldInvalidAddress)?;
  115. match length {
  116. 8 => Ok(context.handler.read_io_u8(port) as u64),
  117. 16 => Ok(context.handler.read_io_u16(port) as u64),
  118. 32 => Ok(context.handler.read_io_u32(port) as u64),
  119. _ => Err(AmlError::FieldInvalidAccessSize),
  120. }
  121. }
  122. RegionSpace::PciConfig => {
  123. /*
  124. * First, we need to get some extra information out of objects in the parent object. Both
  125. * `_SEG` and `_BBN` seem optional, with defaults that line up with legacy PCI implementations
  126. * (e.g. systems with a single segment group and a single root, respectively).
  127. */
  128. let parent_device = self.parent_device.as_ref().unwrap();
  129. let seg = match context.invoke_method(
  130. &AmlName::from_str("_SEG").unwrap().resolve(parent_device).unwrap(),
  131. Args::EMPTY,
  132. ) {
  133. Ok(seg) => seg.as_integer(context)?.try_into().map_err(|_| AmlError::FieldInvalidAddress)?,
  134. Err(AmlError::ValueDoesNotExist(_)) => 0,
  135. Err(err) => return Err(err),
  136. };
  137. let bbn = match context.invoke_method(
  138. &AmlName::from_str("_BBN").unwrap().resolve(parent_device).unwrap(),
  139. Args::EMPTY,
  140. ) {
  141. Ok(bbn) => bbn.as_integer(context)?.try_into().map_err(|_| AmlError::FieldInvalidAddress)?,
  142. Err(AmlError::ValueDoesNotExist(_)) => 0,
  143. Err(err) => return Err(err),
  144. };
  145. let adr = {
  146. let adr = context.invoke_method(
  147. &AmlName::from_str("_ADR").unwrap().resolve(parent_device).unwrap(),
  148. Args::EMPTY,
  149. )?;
  150. adr.as_integer(context)?
  151. };
  152. let device = adr.get_bits(16..24) as u8;
  153. let function = adr.get_bits(0..8) as u8;
  154. let offset = (self.base + offset).try_into().map_err(|_| AmlError::FieldInvalidAddress)?;
  155. match length {
  156. 8 => Ok(context.handler.read_pci_u8(seg, bbn, device, function, offset) as u64),
  157. 16 => Ok(context.handler.read_pci_u16(seg, bbn, device, function, offset) as u64),
  158. 32 => Ok(context.handler.read_pci_u32(seg, bbn, device, function, offset) as u64),
  159. _ => Err(AmlError::FieldInvalidAccessSize),
  160. }
  161. }
  162. // TODO
  163. _ => unimplemented!(),
  164. }
  165. }
  166. /// Perform a standard-size write to this op-region. `length` must be a supported power-of-2,
  167. /// and `offset` correctly aligned for that `length`. `value` must be appropriately sized.
  168. pub fn write(&self, offset: u64, length: u64, value: u64, context: &mut AmlContext) -> Result<(), AmlError> {
  169. match self.region {
  170. RegionSpace::SystemMemory => {
  171. let address = (self.base + offset).try_into().map_err(|_| AmlError::FieldInvalidAddress)?;
  172. match length {
  173. 8 => Ok(context.handler.write_u8(address, value as u8)),
  174. 16 => Ok(context.handler.write_u16(address, value as u16)),
  175. 32 => Ok(context.handler.write_u32(address, value as u32)),
  176. 64 => Ok(context.handler.write_u64(address, value)),
  177. _ => Err(AmlError::FieldInvalidAccessSize),
  178. }
  179. }
  180. RegionSpace::SystemIo => {
  181. let port = (self.base + offset).try_into().map_err(|_| AmlError::FieldInvalidAddress)?;
  182. match length {
  183. 8 => Ok(context.handler.write_io_u8(port, value as u8)),
  184. 16 => Ok(context.handler.write_io_u16(port, value as u16)),
  185. 32 => Ok(context.handler.write_io_u32(port, value as u32)),
  186. _ => Err(AmlError::FieldInvalidAccessSize),
  187. }
  188. }
  189. RegionSpace::PciConfig => {
  190. /*
  191. * First, we need to get some extra information out of objects in the parent object. Both
  192. * `_SEG` and `_BBN` seem optional, with defaults that line up with legacy PCI implementations
  193. * (e.g. systems with a single segment group and a single root, respectively).
  194. */
  195. let parent_device = self.parent_device.as_ref().unwrap();
  196. let seg = match context.invoke_method(
  197. &AmlName::from_str("_SEG").unwrap().resolve(parent_device).unwrap(),
  198. Args::EMPTY,
  199. ) {
  200. Ok(seg) => seg.as_integer(context)?.try_into().map_err(|_| AmlError::FieldInvalidAddress)?,
  201. Err(AmlError::ValueDoesNotExist(_)) => 0,
  202. Err(err) => return Err(err),
  203. };
  204. let bbn = match context.invoke_method(
  205. &AmlName::from_str("_BBN").unwrap().resolve(parent_device).unwrap(),
  206. Args::EMPTY,
  207. ) {
  208. Ok(bbn) => bbn.as_integer(context)?.try_into().map_err(|_| AmlError::FieldInvalidAddress)?,
  209. Err(AmlError::ValueDoesNotExist(_)) => 0,
  210. Err(err) => return Err(err),
  211. };
  212. let adr = {
  213. let adr = context.invoke_method(
  214. &AmlName::from_str("_ADR").unwrap().resolve(parent_device).unwrap(),
  215. Args::EMPTY,
  216. )?;
  217. adr.as_integer(context)?
  218. };
  219. let device = adr.get_bits(16..24) as u8;
  220. let function = adr.get_bits(0..8) as u8;
  221. let offset = (self.base + offset).try_into().map_err(|_| AmlError::FieldInvalidAddress)?;
  222. match length {
  223. 8 => Ok(context.handler.write_pci_u8(seg, bbn, device, function, offset, value as u8)),
  224. 16 => Ok(context.handler.write_pci_u16(seg, bbn, device, function, offset, value as u16)),
  225. 32 => Ok(context.handler.write_pci_u32(seg, bbn, device, function, offset, value as u32)),
  226. _ => Err(AmlError::FieldInvalidAccessSize),
  227. }
  228. }
  229. // TODO
  230. _ => unimplemented!(),
  231. }
  232. }
  233. }
  234. #[derive(Clone, Copy, PartialEq, Eq, Debug)]
  235. pub enum RegionSpace {
  236. SystemMemory,
  237. SystemIo,
  238. PciConfig,
  239. EmbeddedControl,
  240. SMBus,
  241. SystemCmos,
  242. PciBarTarget,
  243. IPMI,
  244. GeneralPurposeIo,
  245. GenericSerialBus,
  246. OemDefined(u8),
  247. }