statement.rs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. use crate::{
  2. opcode::{self, ext_opcode, opcode},
  3. parser::{
  4. choice,
  5. comment_scope,
  6. extract,
  7. id,
  8. take,
  9. take_to_end_of_pkglength,
  10. take_u32,
  11. try_with_context,
  12. ParseResult,
  13. Parser,
  14. Propagate,
  15. },
  16. pkg_length::{pkg_length, PkgLength},
  17. term_object::{term_arg, term_list},
  18. AmlContext,
  19. AmlError,
  20. DebugVerbosity,
  21. };
  22. pub fn statement_opcode<'a, 'c>() -> impl Parser<'a, 'c, ()>
  23. where
  24. 'c: 'a,
  25. {
  26. /*
  27. * StatementOpcode := DefBreak | DefBreakPoint | DefContinue | DefFatal | DefIfElse | DefLoad | DefNoop |
  28. * DefNotify | DefRelease | DefReset | DefReturn | DefSignal | DefSleep | DefStall | DefWhile
  29. */
  30. comment_scope(
  31. DebugVerbosity::AllScopes,
  32. "StatementOpcode",
  33. choice!(
  34. def_break(),
  35. def_breakpoint(),
  36. def_continue(),
  37. def_fatal(),
  38. def_if_else(),
  39. def_noop(),
  40. def_return(),
  41. def_sleep(),
  42. def_stall(),
  43. def_while()
  44. ),
  45. )
  46. }
  47. fn def_break<'a, 'c>() -> impl Parser<'a, 'c, ()>
  48. where
  49. 'c: 'a,
  50. {
  51. /*
  52. * DefBreak := 0xa5
  53. */
  54. opcode(opcode::DEF_BREAK_OP)
  55. .then(comment_scope(
  56. DebugVerbosity::AllScopes,
  57. "DefBreak",
  58. id().map(|()| -> Result<(), Propagate> { Err(Propagate::Break) }),
  59. ))
  60. .discard_result()
  61. }
  62. fn def_breakpoint<'a, 'c>() -> impl Parser<'a, 'c, ()>
  63. where
  64. 'c: 'a,
  65. {
  66. /*
  67. * DefBreakPoint := 0xcc
  68. * TODO: there is no debugger, so this doesn't do anything. If there was, this should stop execution and enter
  69. * the AML debugger.
  70. */
  71. opcode(opcode::DEF_BREAKPOINT_OP)
  72. .then(comment_scope(DebugVerbosity::AllScopes, "DefBreakPoint", id()))
  73. .discard_result()
  74. }
  75. fn def_continue<'a, 'c>() -> impl Parser<'a, 'c, ()>
  76. where
  77. 'c: 'a,
  78. {
  79. /*
  80. * DefContinue := 0x9f
  81. */
  82. opcode(opcode::DEF_CONTINUE_OP)
  83. .then(comment_scope(
  84. DebugVerbosity::AllScopes,
  85. "DefContinue",
  86. id().map(|()| -> Result<(), Propagate> { Err(Propagate::Continue) }),
  87. ))
  88. .discard_result()
  89. }
  90. fn def_fatal<'a, 'c>() -> impl Parser<'a, 'c, ()>
  91. where
  92. 'c: 'a,
  93. {
  94. /*
  95. * DefFatal := ExtOpPrefix 0x32 FatalType FatalCode FatalArg
  96. * FatalType := ByteData
  97. * FatalCode := DWordData
  98. * FatalArg := TermArg => Integer
  99. */
  100. ext_opcode(opcode::EXT_DEF_FATAL_OP)
  101. .then(comment_scope(
  102. DebugVerbosity::Scopes,
  103. "DefFatal",
  104. take().then(take_u32()).then(term_arg()).map_with_context(
  105. |((fatal_type, fatal_code), fatal_arg), context| -> (Result<(), Propagate>, &'c mut AmlContext) {
  106. let fatal_arg = try_with_context!(context, fatal_arg.as_integer(context));
  107. context.handler.handle_fatal_error(fatal_type, fatal_code, fatal_arg);
  108. (Err(Propagate::Err(AmlError::FatalError)), context)
  109. },
  110. ),
  111. ))
  112. .discard_result()
  113. }
  114. fn def_if_else<'a, 'c>() -> impl Parser<'a, 'c, ()>
  115. where
  116. 'c: 'a,
  117. {
  118. /*
  119. * DefIfElse := 0xa0 PkgLength Predicate TermList DefElse
  120. * Predicate := TermArg => Integer (0 = false, >0 = true)
  121. * DefElse := Nothing | <0xa1 PkgLength TermList>
  122. */
  123. let maybe_else_opcode = |input, context| match opcode(opcode::DEF_ELSE_OP).parse(input, context) {
  124. Err((x, y, Propagate::Err(AmlError::UnexpectedEndOfStream))) => {
  125. Err((x, y, Propagate::Err(AmlError::WrongParser)))
  126. }
  127. r => r,
  128. };
  129. opcode(opcode::DEF_IF_ELSE_OP)
  130. .then(comment_scope(
  131. DebugVerbosity::Scopes,
  132. "DefIfElse",
  133. pkg_length()
  134. .then(term_arg())
  135. .feed(|(length, predicate_arg)| {
  136. take_to_end_of_pkglength(length)
  137. .map(move |then_branch| Ok((predicate_arg.as_bool()?, then_branch)))
  138. })
  139. .then(choice!(
  140. maybe_else_opcode
  141. .then(comment_scope(
  142. DebugVerbosity::AllScopes,
  143. "DefElse",
  144. pkg_length().feed(|length| take_to_end_of_pkglength(length)),
  145. ))
  146. .map(|((), else_branch): ((), &[u8])| Ok(else_branch)),
  147. // TODO: can this be `id().map(&[])`?
  148. |input, context| -> ParseResult<'a, 'c, &[u8]> {
  149. /*
  150. * This path parses an DefIfElse that doesn't have an else branch. We simply
  151. * return an empty slice, so if the predicate is false, we don't execute
  152. * anything.
  153. */
  154. Ok((input, context, &[]))
  155. }
  156. ))
  157. .map_with_context(|((predicate, then_branch), else_branch), context| {
  158. let branch = if predicate { then_branch } else { else_branch };
  159. match term_list(PkgLength::from_raw_length(branch, branch.len() as u32).unwrap())
  160. .parse(branch, context)
  161. {
  162. Ok((_, context, result)) => (Ok(result), context),
  163. Err((_, context, err)) => (Err(err), context),
  164. }
  165. }),
  166. ))
  167. .discard_result()
  168. }
  169. fn def_noop<'a, 'c>() -> impl Parser<'a, 'c, ()>
  170. where
  171. 'c: 'a,
  172. {
  173. /*
  174. * DefNoop := 0xa3
  175. */
  176. opcode(opcode::DEF_NOOP_OP).then(comment_scope(DebugVerbosity::AllScopes, "DefNoop", id())).discard_result()
  177. }
  178. fn def_return<'a, 'c>() -> impl Parser<'a, 'c, ()>
  179. where
  180. 'c: 'a,
  181. {
  182. /*
  183. * DefReturn := 0xa4 ArgObject
  184. * ArgObject := TermArg => DataRefObject
  185. */
  186. opcode(opcode::DEF_RETURN_OP)
  187. .then(comment_scope(
  188. DebugVerbosity::Scopes,
  189. "DefReturn",
  190. term_arg().map(|return_arg| -> Result<(), Propagate> {
  191. /*
  192. * To return a value, we want to halt execution of the method and propagate the
  193. * return value all the way up to the start of the method invocation. To do this,
  194. * we emit a special error that is intercepted during method invocation and turned
  195. * into a valid result.
  196. */
  197. Err(Propagate::Return(return_arg))
  198. }),
  199. ))
  200. .discard_result()
  201. }
  202. fn def_sleep<'a, 'c>() -> impl Parser<'a, 'c, ()>
  203. where
  204. 'c: 'a,
  205. {
  206. /*
  207. * DefSleep := ExtOpPrefix 0x22 MSecTime
  208. * MSecTime := TermArg => Integer
  209. */
  210. ext_opcode(opcode::EXT_DEF_SLEEP_OP)
  211. .then(comment_scope(
  212. DebugVerbosity::Scopes,
  213. "DefSleep",
  214. term_arg().map_with_context(|milliseconds, context| {
  215. let milliseconds = try_with_context!(context, milliseconds.as_integer(&context));
  216. context.handler.sleep(milliseconds);
  217. (Ok(()), context)
  218. }),
  219. ))
  220. .discard_result()
  221. }
  222. fn def_stall<'a, 'c>() -> impl Parser<'a, 'c, ()>
  223. where
  224. 'c: 'a,
  225. {
  226. /*
  227. * DefStall := ExtOpPrefix 0x21 USecTime
  228. * USecTime := TermArg => Integer
  229. */
  230. ext_opcode(opcode::EXT_DEF_STALL_OP)
  231. .then(comment_scope(
  232. DebugVerbosity::Scopes,
  233. "DefStall",
  234. term_arg().map_with_context(|microseconds, context| {
  235. let microseconds = try_with_context!(context, microseconds.as_integer(&context));
  236. context.handler.stall(microseconds);
  237. (Ok(()), context)
  238. }),
  239. ))
  240. .discard_result()
  241. }
  242. fn def_while<'a, 'c>() -> impl Parser<'a, 'c, ()>
  243. where
  244. 'c: 'a,
  245. {
  246. /*
  247. * DefWhile := 0xa2 PkgLength Predicate TermList
  248. * Predicate := TermArg => Integer (0 = false, >0 = true)
  249. *
  250. * Parsing this does something a little unusual - it 'extracts' the predicate when it's first parsed, which
  251. * allows us to reevaluate it to see if we should break out of the while yet. This is required, to make sure
  252. * we're observing changes to the context between the iterations of the loop.
  253. */
  254. opcode(opcode::DEF_WHILE_OP)
  255. .then(comment_scope(
  256. DebugVerbosity::Scopes,
  257. "DefWhile",
  258. pkg_length()
  259. .then(extract(term_arg()))
  260. .feed(move |(length, (first_predicate, predicate_stream))| {
  261. take_to_end_of_pkglength(length)
  262. .map(move |body| Ok((first_predicate.clone(), predicate_stream, body)))
  263. })
  264. .map_with_context(|(first_predicate, predicate_stream, body), mut context| {
  265. if !try_with_context!(context, first_predicate.as_bool()) {
  266. return (Ok(()), context);
  267. }
  268. loop {
  269. match term_list(PkgLength::from_raw_length(body, body.len() as u32).unwrap())
  270. .parse(body, context)
  271. {
  272. Ok((_, new_context, _result)) => {
  273. context = new_context;
  274. }
  275. Err((_, new_context, Propagate::Break)) => {
  276. context = new_context;
  277. break;
  278. }
  279. Err((_, new_context, Propagate::Continue)) => {
  280. // We don't need to do anything special here - the `Propagate::Continue` bubbles
  281. // up, and then we can just move on to checking the predicate for the next
  282. // iteration.
  283. context = new_context;
  284. }
  285. Err((_, context, err)) => return (Err(err), context),
  286. }
  287. // Reevaluate the predicate to see if we should break out of the loop yet
  288. let predicate =
  289. match comment_scope(DebugVerbosity::AllScopes, "WhilePredicate", term_arg())
  290. .parse(predicate_stream, context)
  291. {
  292. Ok((_, new_context, result)) => {
  293. context = new_context;
  294. try_with_context!(context, result.as_bool())
  295. }
  296. Err((_, context, err)) => return (Err(err), context),
  297. };
  298. if !predicate {
  299. break;
  300. }
  301. }
  302. (Ok(()), context)
  303. }),
  304. ))
  305. .discard_result()
  306. }