To set up the IFR Assembler, follow the instructions in Part 1.
To learn about the higher-level parsing process of the IFR Assembler, view Part 2.
To learn about the token-parsing process of the IFR Assembler, view Part 3.
This post will walk you through the expression-parsing process of the IFR Assembler.
Understanding Expression Hierarchy
Each expression is a series of operators and operands. Operators are actions that can be performed on values. These actions include addition and subtraction, multiplication and division, and comparisons along with many other actions. Operands are all of the values that the operators act upon.In any mathematical equation, there is an order of precedence. Order of precedence is the order in which the different operators in an equation are executed. This is why multiplication is done before addition in every equation. Expressions in C, like mathematic equations, have their own order of precedence to organize the order in which the operators are processed.
To see the complete order of precedence for the C programming language, click here.
The Expression-Parsing Process
data:image/s3,"s3://crabby-images/c6f59/c6f59fb6cc6f830f4061e1ff72b51e1b49ba3678" alt=""
data:image/s3,"s3://crabby-images/eac15/eac151ac136edb9577ddd5da4bd1deb440398eec" alt=""
data:image/s3,"s3://crabby-images/5dcb2/5dcb28aef586adb8284cc908dfa551ee1c28aa7b" alt=""
calls ParseExpr2(), and ParseExpr2() calls ParseExpr3(), and this process goes on until ParseExpr12().
data:image/s3,"s3://crabby-images/ab1be/ab1beb52895710bf4265a8c798af8535730744d8" alt=""
Table 4.1 - Operands and Operators Parsed by Each Function
data:image/s3,"s3://crabby-images/6b3ab/6b3ab064c25e74b4297408a616080ad28be39410" alt=""
When operators within an expression are parsed, they are turned into their own sub-expression. Depending on the number of operands that the operator requires, a different function is called. For example, if the operator has only one operand, like ++ (T_P__PLUS_PLUS), it is parsed with the function CreateExprUnary().
data:image/s3,"s3://crabby-images/46323/46323b7f57f18bab89dc4332d353e8dc19ba8486" alt=""
data:image/s3,"s3://crabby-images/1272d/1272db66174b1e4d4802bf13fabbab97ebb05ae2" alt=""
data:image/s3,"s3://crabby-images/9323e/9323e75244b39439c4a803ae9dbee138b38b047a" alt=""
If the operands in the expression are unsigned integers (T_P_UINT), CreateExprUint() is called. This function creates an expression from that integer.
data:image/s3,"s3://crabby-images/3df0e/3df0e6e8796cbd2b0786107efae432cb31daa410" alt=""
If the operand is either true (T_P_TRUE) or false (T_P_FALSE), the function CreateExprBoolean() is called to make the expression with a Boolean response.
data:image/s3,"s3://crabby-images/f684d/f684d902373152f283fa031c9756780f7a0e9f32" alt=""
If the operand found turns out to be an IFR opcode, it is treated like a function and function parameters are added to the operand. These operands are found within the parentheses of the opcode and are not written as a separate expression. Instead, their information is recorded as operands of the opcode's expression. A different function is called depending on the number of operands the opcode can hold. If the opcode has a set number of operands, then CreateFn() is called.
data:image/s3,"s3://crabby-images/fbe00/fbe00e3b46997deec99751bd52154e0200d2050c" alt=""
data:image/s3,"s3://crabby-images/208b2/208b2ece497fff3a47853d7e883488917af4ff40" alt=""
The Expression Tree
data:image/s3,"s3://crabby-images/ba70e/ba70e0f576007092825cda4ff3efe45d8e1d34c7" alt=""
data:image/s3,"s3://crabby-images/0818c/0818c3abee1101e06c0cd8b6cbf846cafc28d70e" alt=""
The parser would then point to the asterisk, which serves as the multiplication operator. Since the asterisk is an operator, it has a lower precedence than a, so the multiplication operator is moved up a level. The multiplication operator requires two operands, each one represented by a branch coming off of it in the tree. a is set to one of operands of the multiplication operator.
data:image/s3,"s3://crabby-images/a11d0/a11d02ec8dc14cd477d13470517c5212ac3cc217" alt=""
When b is reached, the parser sees that the multiplication operator still has another required operand, so b is set as that operand.
data:image/s3,"s3://crabby-images/ee813/ee813628c1cc1b4a7f3a8e1f7001d4c7c5706444" alt=""
The parser then moves to the first plus sign, which is the addition operator. Since it is not an operand, the plus sign is moved up a level. Since the plus sign has less precedence than the asterisk, it is moved a level higher than the asterisk. The asterisk is set as one of the operands of the plus sign. The next token, c, is set as the other operand for the plus sign.
data:image/s3,"s3://crabby-images/2c278/2c2780eee1236ce10705d0964a6ff7c9bc1c89d7" alt=""
The parser moves on to the second plus sign. Since the plus sign has less precedence then an operand, the plus sign moves up a level from c. Though both plus signs are on the same level of precedence, precedence goes left to right for plus signs, so the first plus sign takes precedence. The second plus moves a level higher than the first plus sign, becoming the root of the expression tree. The parser then moves onto d, which becomes the second operand of the second plus sign.
Writing Expressions
data:image/s3,"s3://crabby-images/bc6a1/bc6a1cf616b6c7ddb01e009b10e6a84dbc29082c" alt=""
data:image/s3,"s3://crabby-images/bf181/bf18132c31c6c33b068f974927bcd8c38ee13e36" alt=""
WriteExpr() calls WriteExprHelper(), which then takes the expression and creates an opcode out of the expression and its operands. After creating the opcode, the opcode is returned to WriteExpr(), which writes the opcode to the output buffer.
Normally, operands are written to the buffer before the operator is written, so if a + b is found, the opcode, a b + would be written to the output buffer. If an expression does has a specific order for its operators and operands to be written in, an expression-specific function is called. See Table 4.2 to view these expression-specific functions.
Table 4.2 - The table below shows expression-specific expression-writing functions called by WriteExprHelper().
This finishes our series on Writing the IFR Assembler for UEFI. The IFR Assembler used does not several situation. The IFR Assembler does not handle any packages other than form packages nor does it have an optimization process. Also, the IFR Assembler is not free from bugs. This Assembler was simply made to as an example of how to set up an IFR Assembler for UEFI.
Table 4.3 - Operands and Operators Parsed by ParseExpr12()
No comments:
Post a Comment