From ee1e8ee5bdc3c8441551a02c1160d75bc6ed6421 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?St=C3=A9phaneDucasse?= Date: Thu, 8 Aug 2024 19:23:32 +0200 Subject: [PATCH] Playing with ascii for basic execution + typos in method and byte code --- .../3-MethodsAndBytecode/figures/Makefile | 25 + .../3-MethodsAndBytecode/figures/README.md | 36 + .../figures/compile_method_shape-ascii.txt | 14 + .../figures/compile_method_shape.pdf | Bin 0 -> 9375 bytes .../3-MethodsAndBytecode/methodsbytecode.md | 36 +- Chapters/4-Interpreter/figures/README.md | 5 +- .../BasicsOnExecution/basicsOnExecution.md | 737 +++++++++++++++++- .../figures/interpreter_activation-ascii.txt | 7 + .../figures/interpreter_activation.pdf | Bin 0 -> 11768 bytes .../figures/width_0-ascii.txt | 15 +- .../BasicsOnExecution/figures/width_0.pdf | Bin 12622 -> 11748 bytes Chapters/Makefile | 25 + index.md | 6 + 13 files changed, 870 insertions(+), 36 deletions(-) create mode 100644 Chapters/3-MethodsAndBytecode/figures/Makefile create mode 100644 Chapters/3-MethodsAndBytecode/figures/README.md create mode 100644 Chapters/3-MethodsAndBytecode/figures/compile_method_shape-ascii.txt create mode 100644 Chapters/3-MethodsAndBytecode/figures/compile_method_shape.pdf create mode 100644 Chapters/BasicsOnExecution/figures/interpreter_activation-ascii.txt create mode 100644 Chapters/BasicsOnExecution/figures/interpreter_activation.pdf create mode 100644 Chapters/Makefile diff --git a/Chapters/3-MethodsAndBytecode/figures/Makefile b/Chapters/3-MethodsAndBytecode/figures/Makefile new file mode 100644 index 0000000..f440aeb --- /dev/null +++ b/Chapters/3-MethodsAndBytecode/figures/Makefile @@ -0,0 +1,25 @@ +# https://www.gnu.org/software/make/manual/html_node/ +# https://crates.io/crates/svgbob_cli +# https://github.com/ivanceras/svgbob?tab=readme-ov-file + +# Find all the -ascii.txt files we want to compile +# Note the single quotes around the * expressions. The shell will incorrectly expand these otherwise, but we want to send the * directly to the find command. +SRC_DIRS := . +BUILD_DIR := . +SRCS := $(shell find $(SRC_DIRS) -name '*-ascii.txt') +PDFs := $(SRCS:-ascii.txt=.pdf) + +.PHONY: default +default: all + +all: $(PDFs) + +%.svg : %-ascii.txt + svgbob_cli $< -o $@ + +%.pdf : %.svg + rsvg-convert -f pdf -o $@ $< + +.PHONY: clean +clean: + rm -rf *.svg $(PDFs) \ No newline at end of file diff --git a/Chapters/3-MethodsAndBytecode/figures/README.md b/Chapters/3-MethodsAndBytecode/figures/README.md new file mode 100644 index 0000000..b109d8b --- /dev/null +++ b/Chapters/3-MethodsAndBytecode/figures/README.md @@ -0,0 +1,36 @@ +# About the figures + +## Old figures: omnigraffle + +Old figures are made with omnigraffle and exported to pdf and png. +Latex output uses pdf. + +The problem with these is that keeping the same look and feel over all of these is very complicated. + +## New figures: asciiart -> svg -> pdf + +Using asciiart allows me (Guille) to keep the same look and feel in the figures. +Converting ascii art to pdf requires two command line tools: `aasvg` and `rsvg-convert` which I installed as follows in Mac: + +```bash +npm install -g aasvg +sudo apt-get install librsvg2-bin +``` +If the previous apt-get does not work + +Notice that you should also install svgbob + +``` +brew install librsvg +brew install svgbob +``` + +Figure source code is asciiart and named `*-ascii.txt`. +We then convert them to svg using the following commands. + +**TIP: Use the makefile ;)** + +``` +aasvg < x.txt > x.svg +rsvg-convert -f pdf -o x.pdf x.svg +``` \ No newline at end of file diff --git a/Chapters/3-MethodsAndBytecode/figures/compile_method_shape-ascii.txt b/Chapters/3-MethodsAndBytecode/figures/compile_method_shape-ascii.txt new file mode 100644 index 0000000..9cab02e --- /dev/null +++ b/Chapters/3-MethodsAndBytecode/figures/compile_method_shape-ascii.txt @@ -0,0 +1,14 @@ + + +compilemethod -------->+---------------------+ + |'Hello' + |#nextPutAll: + |... +byte code -------> +---------------------+ + |80 + |put real one here + | +method trailer ----> +---------------------+ + | + | +method end ----> +---------------------+ \ No newline at end of file diff --git a/Chapters/3-MethodsAndBytecode/figures/compile_method_shape.pdf b/Chapters/3-MethodsAndBytecode/figures/compile_method_shape.pdf new file mode 100644 index 0000000000000000000000000000000000000000..b03bec0de634dfa0a75e38b69a0cec81046660f8 GIT binary patch literal 9375 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 a&Wj(F3}AbQYwz*`fxH-Wbdriv82> exampleMethod self someComputation > 1 ifTrue: [ ^ #() ] ifFalse: [ self error: 'Unexpected!' ] ``` -In Pharo, literal values are stored each in different a reference slot in a method. +In Pharo, literal values are stored each in different reference slot in a method. The collection of reference slots in a method is called the _literal frame_. Remember from the object representation chapter, that `CompiledMethod` instances are variable objects that contain a variable reference part and a variable byte indexable part. -Commonly, the literal frame contains references to numbers, characters, strings, arrays and symbols used in a method. -In addition, when referencing globals (and thus classes), class variables and shared variables, the literal frame references their corresponding associations. -Finally, the literal frame references also runtime meta-data such as flags or message selectors required to perform message-sends. +- Commonly, the literal frame contains references to numbers, characters, strings, arrays and symbols used in a method. +- In addition, when referencing globals (and thus classes), class variables and shared variables, the literal frame references their corresponding associations. +- Finally, the literal frame references also runtime meta-data such as flags or message selectors required to perform message-sends. It is worth noticing that the literal frame poses no actual limitation to what type of object it references. -Such capability is exploited in rare cases when a method's behavior cannot be expressed in Pharo syntax. -This is for eample the case of foreign function interface methods that are compiled by a separate compiler and stores foreign function meta-data as literals. +Such capability is rarely exploited when a method's behavior cannot be expressed in Pharo syntax. +This is for example the case of foreign function interface methods that are compiled by a separate compiler and store foreign function meta-data as literals. + +### Compiled Method + +A compiled method is structured around a method header, a literal frame, a bytecode sequence and a method trailer as shown in Figure *@methodshape@*. + +![.%anchor=methodshape](figures/compile_method_shape.pdf) #### Method Header diff --git a/Chapters/4-Interpreter/figures/README.md b/Chapters/4-Interpreter/figures/README.md index c1d4686..b109d8b 100644 --- a/Chapters/4-Interpreter/figures/README.md +++ b/Chapters/4-Interpreter/figures/README.md @@ -18,8 +18,11 @@ sudo apt-get install librsvg2-bin ``` If the previous apt-get does not work +Notice that you should also install svgbob + ``` - brew install librsvg +brew install librsvg +brew install svgbob ``` Figure source code is asciiart and named `*-ascii.txt`. diff --git a/Chapters/BasicsOnExecution/basicsOnExecution.md b/Chapters/BasicsOnExecution/basicsOnExecution.md index 71f97ed..b356735 100644 --- a/Chapters/BasicsOnExecution/basicsOnExecution.md +++ b/Chapters/BasicsOnExecution/basicsOnExecution.md @@ -1,25 +1,736 @@ -## Basics on Execution: The interpreter point of view In this chapter we will explain how a compiled method is executed by the virtual machine interpreter. In a future chapter we will explain how the code is compiled by the virtual machine compiler. We will start with the source methods written by programmers. A method is compiled into sequences of instructions called _bytecodes_. The bytecodes produced by the compiler are instructions for an interpreter, which is described in the second section. We will present the logic of interpreting the method bytecodes using a dedicated and internal stack for bytecode. Then we present contexts that support the execution of messages. ### Reminder Imagine the method `Rectangle>>center` defined as follows ``` Rectangle >> width +## Basics on Execution: The interpreter point of view + + +In this chapter we will explain how a compiled method is executed by the virtual machine interpreter. +In a future chapter we will explain how the code is compiled by the virtual machine compiler. + + +We will start with the source methods written by programmers. A method is compiled into sequences of instructions called _bytecodes_. The bytecodes produced by the compiler are instructions for an interpreter, which is described in the second section. We will present the logic of interpreting the method bytecodes using a dedicated and internal stack for bytecode. +Then we present contexts that support the execution of messages. + + +### Reminder + +Imagine the method `Rectangle>>center` defined as follows + +``` +Rectangle >> width "Answer the width of the receiver." - ^ corner x - origin x ``` A programmer does not interact directly with the compiler. When a new source method is added to a class \(`Rectangle` in this example\), the system asks the compiler for an instance of `CompiledMethod` containing the bytecode translation of the source method. The class provides the compiler with some necessary information not given in the source method, including the names of the receiver's instance variables and the dictionaries containing accessible shared variables \(global, class, and pool variables\). The compiler translates the source text into a `CompiledMethod` and the class stores the method in its message dictionary. ### Bytecodes by example Source methods are translated by the compiler into sequences of instructions for a stack-oriented interpreter. The instructions are numbers called bytecodes. For example, the bytecodes corresponding to the source method shown above are: 1, 126, 0, 126, 97, and 92. ``` (Rectangle >> #width) bytecode ->>> #[1 126 0 126 97 92] ``` Pharo supports a simple description of the bytecode using the message `symbolicBytecodes`. ``` (Rectangle >> #width) symbolicBytecodes + ^ corner x - origin x +``` + + +A programmer does not interact directly with the compiler. When a new source method is added to a class (`Rectangle` in this example), the system asks the compiler for an instance of `CompiledMethod` containing the bytecode translation of the source method. +The class provides the compiler with some necessary information not given in the source method, including the names of the receiver's instance variables and the dictionaries containing accessible shared variables (global, class, and pool variables). +The compiler translates the source text into a `CompiledMethod` and the class stores the method in its message dictionary. + +### Bytecodes by example + +Source methods are translated by the compiler into sequences of instructions for a stack-oriented interpreter. +The instructions are numbers called bytecodes. For example, the bytecodes corresponding to the source method shown above are: 1, 126, 0, 126, 97, and 92. + +``` +(Rectangle >> #width) bytecode +>>> #[1 126 0 126 97 92] +``` + + + +Pharo supports a simple description of the bytecode using the message `symbolicBytecodes`. + +``` +(Rectangle >> #width) symbolicBytecodes 25 <01> pushRcvr: 1 26 <7E> send: x 27 <00> pushRcvr: 0 28 <7E> send: x 29 <61> send: - - 30 <5C> returnTop ``` A bytecode's value gives us little indication of its meaning to the interpreter. In this chapter we follow the convention of the Blue Book and accompany lists of bytecodes with comments about their functions. We wrap in parentheses any part of a bytecode's comment that depends on the context of the method in which it appears. The unparenthesized part of the comment describes its general function. For example, the bytecode 0 always instructs the interpreter to push the value of the receiver's first instance variable on to its stack. The fact that the variable is named `origin` depends on the fact that this method is used by the class `Rectangle`, so `origin` is parenthesized. The commented form of the bytecodes for Rectangle center is shown below: Rectangle >> #width - <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack - <7E> send: x - send the unary message x - <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack - <7E> send: x - send the unary message x - <61> send: - - send the binary message - - <5C> returnTop - return the object on top of the stack as the value of the message \(width\) #### About the stack. The stack mentioned in some of the bytecodes is used for several purposes: - In method `width`, it is used to hold the receiver, arguments, and results of the two messages that are sent. - The stack is also used as the source of the result returned from the `width` method. ### A store bytecode Another example of the bytecodes compiled from a source method illustrates the use of a store bytecode. Let us define the method `extent:` as follows: ``` Rectangle >> extent: aPoint - corner := origin + aPoint ``` The message `extent:` changes the receiver's `width` and `height` to be equal to the `x` and `y` coordinates of the argument \(`a Point`\). The receiver's upper left corner \(`origin`\) is kept the same and the lower right corner \(`corner`\) is moved. ``` (Rectangle >> #extent:) bytecode ->>> #[0 64 96 201 88] ``` Rectangle >> #extent: - <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack - <40> pushTemp: 0 - push the argument \(aPoint\) onto the stack - <60> send: `+` - send a binary message + - popIntoRcvr: 1 - pop the top object off of the stack and store it in the receiver's second instance variable \(`corner`\) - 29 <58> returnSelf - return the receiver as the value of the message \(`extent:`\) ### Compiled Methods The compiler creates an instance of `CompiledMethod` to hold the bytecode translation of a source method. In addition to the bytecodes themselves, a `CompiledMethod` contains a set of objects called its _literal frame_. The literal frame contains any objects that could not be referred to directly by bytecodes. All of the objects in the methods `Rectangle>>#extent:` and `Rectangle>>#width` were referred to directly by bytecodes, so the `CompiledMethod` of these methods do not need literal frames. As an example of a `CompiledMethod` with a literal frame, consider the method `Rectangle>>#squishedWithin:`. The `squishedWithin:` message changes the receiver fits within `aRectangle` by reducing its size, not by changing its origin. ``` Rectangle >> squishedWithin: aRectangle + 30 <5C> returnTop +``` + + + +A bytecode's value gives us little indication of its meaning to the interpreter. +In this chapter, we follow the convention of the Blue Book and accompany lists of bytecodes with comments about their functions. + + +We wrap in parentheses any part of a bytecode's comment that depends on the context of the method in which it appears. +The unparenthesized part of the comment describes its general function. +For example, the bytecode 0 always instructs the interpreter to push the value of the receiver's first instance variable onto its stack. +The fact that the variable is named `origin` depends on the fact that this method is used by the class `Rectangle`, so `origin` is parenthesized. +The commented form of the bytecodes for Rectangle width is shown below: + + +![.](figures/width_0.pdf) + +%Rectangle >> #width +%- <01> pushRcvr: 1 - push the value of the receiver's second instance variable (corner) onto the stack +%- <7E> send: x - send the unary message x +%- <00> pushRcvr: 0 - push the value of the receiver's first instance variable (origin) onto the stack +%- <7E> send: x - send the unary message x +%- <61> send: - - send the binary message - +%- <5C> returnTop - return the object on top of the stack as the value of the message (width) + + + +#### About the stack. + +The stack mentioned in some of the bytecodes is used for several purposes: +- In method `width`, it is used to hold the receiver, arguments, and results of the two messages that are sent. +- The stack is also used as the source of the result returned from the `width` method. + + + + +### A store bytecode + + +Another example of the bytecodes compiled from a source method illustrates the use of a store bytecode. +Let us define the method `extent:` as follows: + +``` +Rectangle >> extent: aPoint + corner := origin + aPoint +``` + + +The message `extent:` changes the receiver's `width` and `height` to be equal to the `x` and `y` coordinates of the argument \(`a Point`\). +The receiver's upper left corner \(`origin`\) is kept the same and the lower right corner \(`corner`\) is moved. + +``` +(Rectangle >> #extent:) bytecode +>>> #[0 64 96 201 88] +``` + + + +Rectangle >> #extent: +- <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack +- <40> pushTemp: 0 - push the argument \(aPoint\) onto the stack +- <60> send: `+` - send a binary message + +- popIntoRcvr: 1 - pop the top object off of the stack and store it in the receiver's second instance variable \(`corner`\) +- 29 <58> returnSelf - return the receiver as the value of the message \(`extent:`\) + + + + + + +### Compiled Methods + + +The compiler creates an instance of `CompiledMethod` to hold the bytecode translation of a source method. In addition to the bytecodes themselves, a `CompiledMethod` contains a set of objects called its _literal frame_. +The literal frame contains any objects that could not be referred to directly by bytecodes. + +All of the objects in the methods `Rectangle>>#extent:` and `Rectangle>>#width` were referred to directly by bytecodes, so the `CompiledMethod` of these methods do not need literal frames. + +As an example of a `CompiledMethod` with a literal frame, consider the method `Rectangle>>#squishedWithin:`. +The `squishedWithin:` message changes the receiver fits within `aRectangle` by reducing its size, not by changing its origin. + +``` +Rectangle >> squishedWithin: aRectangle "Return an adjustment of the receiver that fits within aRectangle by reducing its size, not by changing its origin." - ^ origin corner: (corner min: aRectangle bottomRight) ``` ``` (Rectangle >> #squishedWithin:) bytecode ->>> #[0 1 64 128 145 146 92] ``` The message selectors \(`bottomRight`, `min:`, `corner:`\) are not in the set that can be directly referenced by bytecodes. These selectors are included in the `CompiledMethod`'s literal frame and the send bytecodes refer to the selectors by their position in the literal frame. We show the compiledMethod's literal frame after its bytecodes. `Rectangle>>#squishedWithin:` - <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(`origin`\) onto the stack - <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(`corner`\) onto the stack - <40> pushTemp: 0 - push the argument \(`aRectangle`\) - <80> send: bottomRight - send a unary message with the selector in the literal frame location \(`bottomRight`\) - <91> send: min: - send a binary message with the selector in the literal frame location \(`min:`\) - <92> send: corner: - send a keyword message with the selector in the literal frame location \(`corner:`\) - <5C> returnTop - return the object on top of the stack as the value of the message Literal frame - 1 `#bottomRight` - 2 `#min:` - 3 `#corner:` The categories of objects that can be referred to directly by bytecodes are: - the receiver and arguments of the message - the values of the receiver's instance variables - the values of any temporary variables required by the method - special constants \(true, false, nil, -1, 0, 1, and 2\) - special message selectors ``` BytecodeEncoder specialSelectors + ^ origin corner: (corner min: aRectangle bottomRight) +``` + + + +``` +(Rectangle >> #squishedWithin:) bytecode +>>> #[0 1 64 128 145 146 92] +``` + + + +The message selectors \(`bottomRight`, `min:`, `corner:`\) are not in the set that can be directly referenced by bytecodes. +These selectors are included in the `CompiledMethod`'s literal frame and the send bytecodes refer to the selectors by their position in the literal frame. +We show the compiledMethod's literal frame after its bytecodes. + +`Rectangle>>#squishedWithin:` +- <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(`origin`\) onto the stack +- <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(`corner`\) onto the stack +- <40> pushTemp: 0 - push the argument \(`aRectangle`\) +- <80> send: bottomRight - send a unary message with the selector in the literal frame location \(`bottomRight`\) +- <91> send: min: - send a binary message with the selector in the literal frame location \(`min:`\) +- <92> send: corner: - send a keyword message with the selector in the literal frame location \(`corner:`\) +- <5C> returnTop - return the object on top of the stack as the value of the message + +Literal frame +- 1 `#bottomRight` +- 2 `#min:` +- 3 `#corner:` + + +The categories of objects that can be referred to directly by bytecodes are: +- the receiver and arguments of the message +- the values of the receiver's instance variables +- the values of any temporary variables required by the method +- special constants \(true, false, nil, -1, 0, 1, and 2\) +- special message selectors + + +``` +BytecodeEncoder specialSelectors >>> #(#+ #- #< #> #'<=' #'>=' #= #'~=' #* #/ #'\\' #@ #bitShift: #'//' #bitAnd: #bitOr: #at: #at:put: #size #next #nextPut: #atEnd #'==' nil "class" -#'~~' #value #value: #do: #new #new: #x #y) ``` Any objects referred to in a CompiledMethod's bytecodes that do not fall into one of the categories above must appear in its literal frame. The objects ordinarily contained in a literal frame are - shared variables \(global, class, and pool\) - most literal constants \(numbers, characters, strings, arrays, and symbols\) - most message selectors \(those that are not special\) Objects of these three types may be intermixed in the literal frame. % % we could add the following as in the blue book % Two types of object that were referred to above, temporary variables and shared variables, have not been used in the example methods. The following example method for Rectangle merge: uses both types. The merge: message is used to find a Rectangle that includes the areas in both the receiver and the argument. % merge: aRectangle % | minPoint maxPoint | % minPoint := origin min: aRectangle origin. % maxPoint := corner max: aRectangle corner. % ^Rectangle origin: minPoint corner: maxPoint % When a CompiledMethod uses temporary variables (maxPoint and minPoint in this example), the number required is specified in %the first line of its printed form. When a CompiledMethod uses a shared variable (Rectangle in this example) an instance of Association is included in its literal frame. All CompiledMethods that refer to a particular shared variable's name include the same Association in their literal frames. % Rectangle merge: requires 2 temporary variables % 0 push the value of the receiver's first instance variable (origin) onto the stack % 16 push the contents of the first temporary frame location (the argument aRectangle) onto the stack % 209 send a unary message with the selector in the second literal frame location (origin) % 224 send the single argument message with the selector in the first literal frame location (min:) % 105 pop the top object off of the stack and stare in the second temporary frame location (minPoint) % 1 push the value of the receiver's second instance variable (corner) onto the stack % 16 push the contents of the first temporary frame location (the argument aRectangle) onto the stack % 211 send a unary message with the selector in the fourth literal frame location (corner) % 226 send a single argument message with the selector in the third literal frame location (max:) % 106 pop the top object off of the stack and store it in the third temporary frame location (maxPoint) % 69 push the value of the shared variable in the sixth literal frame location (Rectangle) onto the stack % 17 push the contents of the second temporary frame location (minPoint) onto the stack % 18 push the contents of the third temporary frame location (maxPoint) onto the stack % 244 send the two argument message with the selector in the fifth literal frame location (origin:corner:) % 124 return the object on top of the stack as the value of the message (merge:) % literal frame % #min: % #origin % #max: % #corner % #origin:corner: % Association: #Rectangle -> Rectangle ### Temporary Variables Temporary variables are created for a particular execution of a CompiledMethod and cease to exist when the execution is complete. The CompiledMethod indicates to the interpreter how many temporary variables are required. The arguments of the message and the values of the temporary variables are stored together in the temporary frame. The arguments are stored first and the temporary variable values immediately after. They are accessed by the same type of bytecode \(whose comments refer to a temporary frame location\). The compiler enforces the fact that the values of the argument names cannot be changed by never issuing a store bytecode referring to the part of the temporary frame inhabited by the arguments. #### Shared Variables. Shared variables are held in dictionaries or environments. - global variables in the system dictionary unique instance. - class variables in a dictionary held in the class declaring them. - pool variables in collection held in shared pool classes. The system represents associations in general, and shared variables in particular, with instances of `Association` \(representing key value pair of information\). When the compiler encounters the name of a shared variable in a source method, the `Association` with the same name is included in the CompiledMethod's literal frame. The bytecodes that access shared variables indicate the location of an `Association` in the literal frame. The actual value of the variable is stored in an instance variable of the `Association`. Note that the literal frame or dictionaries are holding an `Association` and not just the value because this way it does not force a recompilation of all the users of a shared variable when its value change. ### The bytecodes The interpreter understands bytecode instructions that fall into five categories: pushes, stores, sends, returns, and jumps. This section gives a general description of each type of bytecode without going into detail about which bytecode represents which instruction. Some of the bytecodes take extensions. An extension is one or two bytes following the bytecode that further specify the instruction. An extension is not an instruction on its own, it is only a part of an instruction. #### Push Bytecodes. A push bytecode indicates the source of an object to be added to the top of the interpreter's stack. The sources include - the receiver of the message that invoked the CompiledMethod - the instance variables of the receiver - the temporary frame \(the arguments of the message and the temporary variables\) - the literal frame of the CompiledMethod - the top of the stack \(i.e., this bytecode duplicates the top of stack\) Examples of most of the types of push bytecode have been included in the examples. The bytecode that duplicates the top of the stack is used to implement cascaded messages. Two different types of push bytecode use the literal frame as their source. One is used to push literal constants and the other to push the value of shared variables. Literal constants are stored directly in the literal frame, but the values of shared variables are stored in an Association that is pointed to by the literal frame. % The following example method uses one shared variable and one literal constant. % incrementIndex % ^Index := Index + 4 % ExampleClass incrementIndex % 64 push the value of the shared variable in the first literal frame location (Index) onto the stack % 33 push the constant in the second literal frame location (4) onto the stack % 176 send a binary message with the selector + % 129,192 store the object on top of the stack in the shared variable in the first literal frame location (Index) % 124 return the object on top of the stack as the value of the message (incrementIndex) % literal frame % Association: #Index -> 260 #### Store Bytecodes. The bytecodes compiled from an assignment expression end with a store bytecode. The bytecodes before the store bytecode compute the new value of a variable and leave it on top of the stack. A store bytecode indicates the variable whose value should be changed. The variables that can be changed are - the instance variables of the receiver - temporary variables - shared variables Some of the store bytecodes remove the object to be stored from the stack, and others leave the object on top of the stack, after storing it. #### Send Bytecodes. A send bytecode specifies the selector of a message to be sent and how many arguments it should have. The receiver and arguments of the message are taken off the interpreter's stack, the receiver from below the arguments. By the time the bytecode following the send is executed, the message's result will have replaced its receiver and arguments on the top of the stack. The details of sending messages and returning results is the subject of the next sections of this chapter. A set of 32 send bytecodes refer directly to the special selectors listed earlier. The other send bytecodes refer to their selectors in the literal frame. #### Return Bytecodes. When a return bytecode is encountered, the CompiledMethod in which it was found has been completely executed. Therefore a value is returned for the message that invoked that CompiledMethod. The value is usually found on top of the stack. Four special return bytecodes return the message receiver \(self\), true, false, and nil. #### Jump Bytecodes. Ordinarily, the interpreter executes the bytecodes sequentially in the order they appear in a CompiledMethod. The jump bytecodes indicate that the next bytecode to execute is not the one following the jump. There are two varieties of jump, unconditional and conditional. The unconditional jumps transfer control whenever they are encountered. The conditional jumps will only transfer control if the top of the stack is a specified value. Some of the conditional jumps transfer if the top object on the stack is true and others if it is false. The jump bytecodes are used to implement efficient control structures. The control structures that are so optimized by the compiler are conditional selection messages to Booleans \(`ifTrue:`, `ifFalse:`, and `ifTrue:ifFalse:`\), some of the logical operation messages to Booleans \(`and:` and `or:`\), and the conditional repetition messages to blocks \(`whileTrue:` and `whileFalse:`\). The jump bytecodes indicate the next bytecode to be executed relative to the position of the jump. In other words, they tell the interpreter how many bytecodes to skip. % The following method for Rectangle includesPoint: uses a conditional jump. % includesPoint: aPoint % origin <= aPoint % ifTrue: [^aPoint < corner] % ifFalse: [^false] % Rectangle includesPoint: % 0 push the value of the receiver's first instance variable (origin) onto the stack % 16 push the contents of the first temporary frame location (the argument aPoint) onto the stack % 180 send a binary message with the selector <= % 155 jump ahead 4 bytecodes if the object on top of the stack is false % 16 push the contents of the first temporary frame location (the argument aPoint) onto the stack % 1 push the value of the receiver's second instance variable (corner) onto the stack % 178 send a binary message with the selector < % 124 return the object on top of the stack as the value of the message ( includesPoint:) % 122 return false as the value of the message (includesPoint:) ### The Interpreter The virtual machine contains one interpreter and a compiler \(that compiles to assembly code\). Here we talk about the interpreter. It executes the bytecode instructions found in CompiledMethods. The interpreter uses five pieces of information refered hereafter as the state of the interpreter and repeatedly performs a three-step cycle. #### The State of the Interpreter. - The CompiledMethod whose bytecodes are being executed. - The location of the next bytecode to be executed in that CompiledMethod. This is the interpreter's _instruction pointer_. - The receiver and arguments of the message that invoked the CompiledMethod. - Any temporary variables needed by the CompiledMethod. - A stack. The execution of most bytecodes involves the interpreter's stack. Push bytecodes tell where to find objects to add to the stack. Store bytecodes tell where to put objects found on the stack. Send bytecodes remove the receiver and arguments of messages from the stack. When the result of a message is computed, it is pushed onto the stack. #### The Cycle of the Interpreter. - Fetch the bytecode from the CompiledMethod indicated by the instruction pointer. - Increment the instruction pointer. - Perform the function specified by the bytecode. As an example of the interpreter's function, we will trace its execution of the CompiledMethod `Rectangle>>#width`. The state of the interpreter will be displayed after each of its cycles. The instruction pointer will be indicated by an arrow pointing at the next bytecode in the CompiledMethod to be executed. - **==>** <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack The receiver, arguments, temporary variables, and objects on the stack will be shown as normally printed. For example, if a message is sent to a Rectangle, the receiver will be shown as: - **Receiver:** 50@50 corner: 200@200 At the start of execution, the stack is empty and the instruction pointer indicates the first bytecode in the CompiledMethod. This CompiledMethod does not require temporaries and the invoking message did not have arguments, so these two categories are also empty. #### Step 1. The interpreter is in an initial state. It points to the next instructions to be executed. It knows the receiver of the method to be executed. Rectangle >> #width - **==>** <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack - <7E> send: x - send the unaay message x - <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack - <7E> send: x - send the unary message x - <61> send: - - send the binary message - - <5C> returnTop - return the object on top of the stack as the value of the message \(width\) - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** #### Step 2. Following one cycle of the interpreter, the value of the receiver's second instance variable has been copied onto the stack and the instruction pointer has been advanced. Rectangle >> #width - <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack - **==>** <7E> send: x - send the unray message with the selector x - <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack - <7E> send: x - send the unary message with the selector x - <61> send: - - send the binary message with the selector - - <5C> returnTop - return the object on top of the stack as the value of the message \(width\) - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** 200@200 #### Step 3. The interpreter encounters a send bytecode. It removes one object from the stack and uses it as the receiver of a message with selector `x`. Sending a message will not be described in detail here. Sending messages will be described in later sections. For the moment, it is only necessary to know that eventually the result of the `x` message will be pushed onto the stack. Rectangle >> #width - <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack - <7E> send: x - send the unray message with the selector x - **==>** <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack - <7E> send: x - send the unary message with the selector x - <61> send: - - send the binary message with the selector - - <5C> returnTop - return the object on top of the stack as the value of the message \(width\) - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** 200 #### Step 4. In this cycle, the value of the first receiver's first instance variables \(origin\) onto the stack Rectangle >> #width - <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack - <7E> send: x - send the unray message with the selector x - <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack - **==>** <7E> send: x - send the unary message with the selector x - <61> send: - - send the binary message with the selector - - <5C> returnTop - return the object on top of the stack as the value of the message \(width\) - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** 200 - **Stack:** 50@50 #### Step 5. In this cycle, the message x is sent: it removes one object from the stack and uses it as the receiver of a message with selector `x` and push back the result on the stack. Rectangle >> #width - <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack - <7E> send: x - send the unray message with the selector x - <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack - <7E> send: x - send the unary message with the selector x - **==>** <61> send: - - send the binary message with the selector - - <5C> returnTop - return the object on top of the stack as the value of the message \(width\) - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** 200 - **Stack:** 50 #### Step 6. The final bytecode returns a result to the width message. The result is found on the stack \(150\). It is clear by this point that a return bytecode must involve pushing the result onto another stack. The details of returning a value to a message will be described after the description of sending a message. Rectangle >> #width - <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack - <7E> send: x - send the unray message with the selector x - <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack - <7E> send: x - send the unary message with the selector x - <61> send: - - send the binary message with the selector - - **==>** <5C> returnTop - return the object on top of the stack as the value of the message \(width\) - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** 150 ### Contexts Push, store, and jump bytecodes require only small changes to the state of the interpreter. Objects may be moved to or from the stack, and the instruction pointer is always changed; but most of the state remains the same. Send and return bytecodes require much larger changes to the interpreter's state. When a new message is sent \(for example message `x` or `-` above\), all five parts of the interpreter's state may have to be changed to execute a different CompiledMethod in response to this new message. The interpreter's old state must be remembered because the bytecodes after the send must be executed after the value of the message is returned. The interpreter saves its state in objects called _contexts_. There will be many contexts in the system at any one time. The context that represents the current state of the interpreter is called the _active_ context. When a send bytecode in the active context's CompiledMethod requires a new compiled method to be executed, the active context becomes _suspended_ and a new context is created and made active. The suspended context retains the state associated with the original compiled method until that context becomes active again. A context must remember the context that it suspended so that the suspended context can be resumed when a result is returned. The suspended context is called the new context's _sender_. We extend the form used to show the interpreter's state. The active context will be indicated by the word **Active** in its top delimiter. Suspended contexts will say **Suspended**. ### Example For example, consider a context representing the execution of the compiled method `Rectangle>>#rightCenter` with a receiver of `50@50 corner: 200@200`. The sender is some other context in the system. The source method is: ``` Rectangle >> rightCenter - ^self right @ self center y ``` #### After the execution of the first bytecode The following active context shows that state of the context after the execution of the first bytecode: **Active** Rectangle >> #rightCenter - <4C> self - push the receiver \(self\) onto the stack - **==>** <80> send: right - send a unary message with the selector in the first literal \(right\) - <4C> self push the receiver \(self\) onto the stack - <81> send: center - send a unary message with the selector in the first literal \(center\) - <7F> send: y - send a unary message with the selector in the first literal \(y\) - <6B> send: @ - send a unary message with the selector in the first literal \(@\) - <5C> returnTop - return the object on top of the stack as the value of the message \(rightCenter\) Literal frame - 1 `#right` - 2 `#center` - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack** 50@50 corner: 200@200 #### Execution of the second bytecode After the next bytecode is executed, that context will be suspended. The object pushed by the first bytecode has been removed to be used as the receiver of a new context, which becomes active. The new active context is shown above the suspended context for the method `right`: ``` Rectangle >> right +#'~~' #value #value: #do: #new #new: #x #y) +``` + + + + Any objects referred to in a CompiledMethod's bytecodes that do not fall into one of the categories above must appear in its literal frame. The objects ordinarily contained in a literal frame are +- shared variables \(global, class, and pool\) +- most literal constants \(numbers, characters, strings, arrays, and symbols\) +- most message selectors \(those that are not special\) + +Objects of these three types may be intermixed in the literal frame. + + +% % we could add the following as in the blue book +% Two types of object that were referred to above, temporary variables and shared variables, have not been used in the example methods. The following example method for Rectangle merge: uses both types. The merge: message is used to find a Rectangle that includes the areas in both the receiver and the argument. +% merge: aRectangle +% | minPoint maxPoint | +% minPoint := origin min: aRectangle origin. +% maxPoint := corner max: aRectangle corner. +% ^Rectangle origin: minPoint corner: maxPoint +% When a CompiledMethod uses temporary variables (maxPoint and minPoint in this example), the number required is specified in %the first line of its printed form. When a CompiledMethod uses a shared variable (Rectangle in this example) an instance of Association is included in its literal frame. All CompiledMethods that refer to a particular shared variable's name include the same Association in their literal frames. + +% Rectangle merge: requires 2 temporary variables +% 0 push the value of the receiver's first instance variable (origin) onto the stack +% 16 push the contents of the first temporary frame location (the argument aRectangle) onto the stack +% 209 send a unary message with the selector in the second literal frame location (origin) +% 224 send the single argument message with the selector in the first literal frame location (min:) +% 105 pop the top object off of the stack and stare in the second temporary frame location (minPoint) +% 1 push the value of the receiver's second instance variable (corner) onto the stack +% 16 push the contents of the first temporary frame location (the argument aRectangle) onto the stack +% 211 send a unary message with the selector in the fourth literal frame location (corner) +% 226 send a single argument message with the selector in the third literal frame location (max:) +% 106 pop the top object off of the stack and store it in the third temporary frame location (maxPoint) +% 69 push the value of the shared variable in the sixth literal frame location (Rectangle) onto the stack +% 17 push the contents of the second temporary frame location (minPoint) onto the stack +% 18 push the contents of the third temporary frame location (maxPoint) onto the stack +% 244 send the two argument message with the selector in the fifth literal frame location (origin:corner:) +% 124 return the object on top of the stack as the value of the message (merge:) +% literal frame +% #min: +% #origin +% #max: +% #corner +% #origin:corner: +% Association: #Rectangle -> Rectangle + +### Temporary Variables + +Temporary variables are created for a particular execution of a CompiledMethod and cease to exist when the execution is complete. +The CompiledMethod indicates to the interpreter how many temporary variables are required. +The arguments of the message and the values of the temporary variables are stored together in the temporary frame. +The arguments are stored first and the temporary variable values immediately after. +They are accessed by the same type of bytecode \(whose comments refer to a temporary frame location\). + +The compiler enforces the fact that the values of the argument names cannot be changed by never issuing a store bytecode referring to the part of the temporary frame inhabited by the arguments. + +#### Shared Variables. + +Shared variables are held in dictionaries or environments. + +- global variables in the system dictionary unique instance. +- class variables in a dictionary held in the class declaring them. +- pool variables in collection held in shared pool classes. + + +The system represents associations in general, and shared variables in particular, with instances of `Association` \(representing key value pair of information\). +When the compiler encounters the name of a shared variable in a source method, the `Association` with the same name is included in the CompiledMethod's literal frame. +The bytecodes that access shared variables indicate the location of an `Association` in the literal frame. +The actual value of the variable is stored in an instance variable of the `Association`. + +Note that the literal frame or dictionaries are holding an `Association` and not just the value because this way it does not force a recompilation of all the users of a shared variable when its value change. + + + +### The bytecodes + + +The interpreter understands bytecode instructions that fall into five categories: pushes, stores, sends, returns, and jumps. This section gives a general description of each type of bytecode without going into detail about which bytecode represents which instruction. + +Some of the bytecodes take extensions. An extension is one or two bytes following the bytecode that further specify the instruction. An extension is not an instruction on its own, it is only a part of an instruction. + +#### Push Bytecodes. + + A push bytecode indicates the source of an object to be added to the top of the interpreter's stack. The sources include + +- the receiver of the message that invoked the CompiledMethod +- the instance variables of the receiver +- the temporary frame \(the arguments of the message and the temporary variables\) +- the literal frame of the CompiledMethod +- the top of the stack \(i.e., this bytecode duplicates the top of stack\) + + +Examples of most of the types of push bytecode have been included in the examples. +The bytecode that duplicates the top of the stack is used to implement cascaded messages. + +Two different types of push bytecode use the literal frame as their source. +One is used to push literal constants and the other to push the value of shared variables. +Literal constants are stored directly in the literal frame, but the values of shared variables are stored in an Association that is pointed to by the literal frame. + + +% The following example method uses one shared variable and one literal constant. +% incrementIndex +% ^Index := Index + 4 + +% ExampleClass incrementIndex +% 64 push the value of the shared variable in the first literal frame location (Index) onto the stack +% 33 push the constant in the second literal frame location (4) onto the stack +% 176 send a binary message with the selector + +% 129,192 store the object on top of the stack in the shared variable in the first literal frame location (Index) +% 124 return the object on top of the stack as the value of the message (incrementIndex) +% literal frame +% Association: #Index -> 260 + +#### Store Bytecodes. + + +The bytecodes compiled from an assignment expression end with a store bytecode. The bytecodes before the store bytecode compute the new value of a variable and leave it on top of the stack. A store bytecode indicates the variable whose value should be changed. The variables that can be changed are +- the instance variables of the receiver +- temporary variables +- shared variables + + +Some of the store bytecodes remove the object to be stored from the stack, and others leave the object on top of the stack, after storing it. + + +#### Send Bytecodes. + +A send bytecode specifies the selector of a message to be sent and how many arguments it should have. The receiver and arguments of the message are taken off the interpreter's stack, the receiver from below the arguments. By the time the bytecode following the send is executed, the message's result will have replaced its receiver and arguments on the top of the stack. The details of sending messages and returning results is the subject of the next sections of this chapter. A set of 32 send bytecodes refer directly to the special selectors listed earlier. The other send bytecodes refer to their selectors in the literal frame. + +#### Return Bytecodes. + +When a return bytecode is encountered, the CompiledMethod in which it was found has been completely executed. Therefore a value is returned for the message that invoked that CompiledMethod. The value is usually found on top of the stack. Four special return bytecodes return the message receiver \(self\), true, false, and nil. + +#### Jump Bytecodes. + + +Ordinarily, the interpreter executes the bytecodes sequentially in the order they appear in a CompiledMethod. The jump bytecodes indicate that the next bytecode to execute is not the one following the jump. There are two varieties of jump, unconditional and conditional. The unconditional jumps transfer control whenever they are encountered. The conditional jumps will only transfer control if the top of the stack is a specified value. Some of the conditional jumps transfer if the top object on the stack is true and others if it is false. The jump bytecodes are used to implement efficient control structures. + + +The control structures that are so optimized by the compiler are conditional selection messages to Booleans \(`ifTrue:`, `ifFalse:`, and `ifTrue:ifFalse:`\), some of the logical operation messages to Booleans \(`and:` and `or:`\), and the conditional repetition messages to blocks \(`whileTrue:` and `whileFalse:`\). The jump bytecodes indicate the next bytecode to be executed relative to the position of the jump. In other words, they tell the interpreter how many bytecodes to skip. + +% The following method for Rectangle includesPoint: uses a conditional jump. +% includesPoint: aPoint +% origin <= aPoint +% ifTrue: [^aPoint < corner] +% ifFalse: [^false] +% Rectangle includesPoint: +% 0 push the value of the receiver's first instance variable (origin) onto the stack +% 16 push the contents of the first temporary frame location (the argument aPoint) onto the stack +% 180 send a binary message with the selector <= +% 155 jump ahead 4 bytecodes if the object on top of the stack is false +% 16 push the contents of the first temporary frame location (the argument aPoint) onto the stack +% 1 push the value of the receiver's second instance variable (corner) onto the stack +% 178 send a binary message with the selector < +% 124 return the object on top of the stack as the value of the message ( includesPoint:) +% 122 return false as the value of the message (includesPoint:) + + + + + + + + + + + + +### The Interpreter + + +The virtual machine contains one interpreter and a compiler \(that compiles to assembly code\). Here we talk about the interpreter. It executes the bytecode instructions found in CompiledMethods. The interpreter uses five pieces of information refered hereafter as the state of the interpreter and repeatedly performs a three-step cycle. + +#### The State of the Interpreter. + +- The CompiledMethod whose bytecodes are being executed. +- The location of the next bytecode to be executed in that CompiledMethod. This is the interpreter's _instruction pointer_. +- The receiver and arguments of the message that invoked the CompiledMethod. +- Any temporary variables needed by the CompiledMethod. +- A stack. + + +The execution of most bytecodes involves the interpreter's stack. Push bytecodes tell where to find objects to add to the stack. Store bytecodes tell where to put objects found on the stack. Send bytecodes remove the receiver and arguments of messages from the stack. When the result of a message is computed, it is pushed onto the stack. + +#### The Cycle of the Interpreter. + +- Fetch the bytecode from the CompiledMethod indicated by the instruction pointer. +- Increment the instruction pointer. +- Perform the function specified by the bytecode. + + +As an example of the interpreter's function, we will trace its execution of the CompiledMethod `Rectangle>>#width`. +The state of the interpreter will be displayed after each of its cycles. +The instruction pointer will be indicated by an arrow pointing at the next bytecode in the CompiledMethod to be executed. + +- **==>** <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack + + + +The receiver, arguments, temporary variables, and objects on the stack will be shown as normally printed. +For example, if a message is sent to a Rectangle, the receiver will be shown as: + +- **Receiver:** 50@50 corner: 200@200 + + + +At the start of execution, the stack is empty and the instruction pointer indicates the first bytecode in the CompiledMethod. This CompiledMethod does not require temporaries and the invoking message did not have arguments, so these two categories are also empty. + +#### Step 1. + +The interpreter is in an initial state. It points to the next instructions to be executed. It knows the receiver of the method to be executed. + +Rectangle >> #width +- **==>** <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack +- <7E> send: x - send the unaay message x +- <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack +- <7E> send: x - send the unary message x +- <61> send: - - send the binary message - +- <5C> returnTop - return the object on top of the stack as the value of the message \(width\) +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** + + +#### Step 2. + +Following one cycle of the interpreter, the value of the receiver's second instance variable has been copied onto the stack and the instruction pointer has been advanced. + +Rectangle >> #width +- <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack +- **==>** <7E> send: x - send the unray message with the selector x +- <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack +- <7E> send: x - send the unary message with the selector x +- <61> send: - - send the binary message with the selector - +- <5C> returnTop - return the object on top of the stack as the value of the message \(width\) +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** 200@200 + + + +#### Step 3. + +The interpreter encounters a send bytecode. +It removes one object from the stack and uses it as the receiver of a message with selector `x`. +Sending a message will not be described in detail here. +Sending messages will be described in later sections. +For the moment, it is only necessary to know that eventually the result of the `x` message will be pushed onto the stack. + +Rectangle >> #width +- <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack +- <7E> send: x - send the unray message with the selector x +- **==>** <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack +- <7E> send: x - send the unary message with the selector x +- <61> send: - - send the binary message with the selector - +- <5C> returnTop - return the object on top of the stack as the value of the message \(width\) +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** 200 + + +#### Step 4. + +In this cycle, the value of the first receiver's first instance variables \(origin\) onto the stack + +Rectangle >> #width +- <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack +- <7E> send: x - send the unray message with the selector x +- <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack +- **==>** <7E> send: x - send the unary message with the selector x +- <61> send: - - send the binary message with the selector - +- <5C> returnTop - return the object on top of the stack as the value of the message \(width\) +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** 200 +- **Stack:** 50@50 + + + +#### Step 5. + +In this cycle, the message x is sent: it removes one object from the stack and uses it as the receiver of a message with selector `x` and push back the result on the stack. + +Rectangle >> #width +- <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack +- <7E> send: x - send the unray message with the selector x +- <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack +- <7E> send: x - send the unary message with the selector x +- **==>** <61> send: - - send the binary message with the selector - +- <5C> returnTop - return the object on top of the stack as the value of the message \(width\) +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** 200 +- **Stack:** 50 + + +#### Step 6. + +The final bytecode returns a result to the width message. The result is found on the stack \(150\). It is clear by this point that a return bytecode must involve pushing the result onto another stack. +The details of returning a value to a message will be described after the description of sending a message. + + +Rectangle >> #width +- <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack +- <7E> send: x - send the unray message with the selector x +- <00> pushRcvr: 0 - push the value of the receiver's first instance variable \(origin\) onto the stack +- <7E> send: x - send the unary message with the selector x +- <61> send: - - send the binary message with the selector - +- **==>** <5C> returnTop - return the object on top of the stack as the value of the message \(width\) +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** 150 + + + +### Contexts + + +Push, store, and jump bytecodes require only small changes to the state of the interpreter. +Objects may be moved to or from the stack, and the instruction pointer is always changed; but most of the state remains the same. Send and return bytecodes require much larger changes to the interpreter's state. + +When a new message is sent \(for example message `x` or `-` above\), all five parts of the interpreter's state may have to be changed to execute a different CompiledMethod in response to this new message. The interpreter's old state must be remembered because the bytecodes after the send must be executed after the value of the message is returned. + +The interpreter saves its state in objects called _contexts_. There will be many contexts in the system at any one time. The context that represents the current state of the interpreter is called the _active_ context. When a send bytecode in the active context's CompiledMethod requires a new compiled method to be executed, the active context becomes _suspended_ and a new context is created and made active. The suspended context retains the state associated with the original compiled method until that context becomes active again. A context must remember the context that it suspended so that the suspended context can be resumed when a result is returned. The suspended context is called the new context's _sender_. + + + +We extend the form used to show the interpreter's state. The active context will be indicated by the word **Active** in its top delimiter. Suspended contexts will say **Suspended**. + + +### Example + + +For example, consider a context representing the execution of the compiled method `Rectangle>>#rightCenter` with a receiver of `50@50 corner: 200@200`. The sender is some other context in the system. The source method is: + +``` +Rectangle >> rightCenter + ^self right @ self center y +``` + + + +#### After the execution of the first bytecode + +The following active context shows that state of the context after the execution of the first bytecode: + +**Active** Rectangle >> #rightCenter +- <4C> self - push the receiver \(self\) onto the stack +- **==>** <80> send: right - send a unary message with the selector in the first literal \(right\) +- <4C> self push the receiver \(self\) onto the stack +- <81> send: center - send a unary message with the selector in the first literal \(center\) +- <7F> send: y - send a unary message with the selector in the first literal \(y\) +- <6B> send: @ - send a unary message with the selector in the first literal \(@\) +- <5C> returnTop - return the object on top of the stack as the value of the message \(rightCenter\) + +Literal frame +- 1 `#right` +- 2 `#center` +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack** 50@50 corner: 200@200 + + + +#### Execution of the second bytecode + + +After the next bytecode is executed, that context will be suspended. The object pushed by the first bytecode has been removed to be used as the receiver of a new context, which becomes active. The new active context is shown above the suspended context for the method `right`: + +``` +Rectangle >> right "Answer the position of the receiver's right vertical line." - ^ corner x ``` **Active** Rectangle >> #right - **==>**<01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack - <7E> send: x - <5C> returnTop Literal frame - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** **Suspended** Rectangle >> #rightCenter - <4C> self - push the receiver \(self\) onto the stack - <80> send: right - send a unary message with the selector in the first literal \(right\) - **==>** <4C> self push the receiver \(self\) onto the stack - <81> send: center - send a unary message with the selector in the first literal \(center\) - <7F> send: y - send a unary message with the selector in the first literal \(y\) - <6B> send: @ - send a unary message with the selector in the first literal \(@\) - <5C> returnTop - return the object on top of the stack as the value of the message \(rightCenter\) Literal frame - 1 `#right` - 2 `#center` - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** #### Execution of the right message The next cycle of the interpreter advances the new context instead of the previous one. **Active** Rectangle >> #right - <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack - **==>**<7E> send: x - <5C> returnTop Literal frame - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** 200@200 **Suspended** Rectangle >> #rightCenter - <4C> self - push the receiver \(self\) onto the stack - <80> send: right - send a unary message with the selector in the first literal \(right\) - **==>** <4C> self push the receiver \(self\) onto the stack - <81> send: center - send a unary message with the selector in the first literal \(center\) - <7F> send: y - send a unary message with the selector in the first literal \(y\) - <6B> send: @ - send a unary message with the selector in the first literal \(@\) - <5C> returnTop - return the object on top of the stack as the value of the message \(rightCenter\) Literal frame - 1 `#right` - 2 `#center` - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** #### Execution of the right message In the next cycle, another message is sent, perhaps creating another context. Instead of following the response of this new message \(`x`\), we will skip to the point that this context returns a value \(to `right`\). When the result of `x` has been returned, the new context looks like this: **Active** Rectangle >> #right - <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack - <7E> send: x - **==>** <5C> returnTop Literal frame - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** 200 **Suspended** Rectangle >> #rightCenter - <4C> self - push the receiver \(self\) onto the stack - <80> send: right - send a unary message with the selector in the first literal \(right\) - **==>** <4C> self push the receiver \(self\) onto the stack - <81> send: center - send a unary message with the selector in the first literal \(center\) - <7F> send: y - send a unary message with the selector in the first literal \(y\) - <6B> send: @ - send a unary message with the selector in the first literal \(@\) - <5C> returnTop - return the object on top of the stack as the value of the message \(rightCenter\) Literal frame - 1 `#right` - 2 `#center` - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** #### Returning from right The next bytecode returns the value \(200\) on the top of the active context's stack as the value of the message that created the context \(`right`\). The active context's sender becomes the active context again and the returned value is pushed on its stack. It is ready to continue its execution. **Active** Rectangle >> #rightCenter - <4C> self - push the receiver \(self\) onto the stack - <80> send: right - send a unary message with the selector in the first literal \(right\) - **==>** <4C> self push the receiver \(self\) onto the stack - <81> send: center - send a unary message with the selector in the first literal \(center\) - <7F> send: y - send a unary message with the selector in the first literal \(y\) - <6B> send: @ - send a unary message with the selector in the first literal \(@\) - <5C> returnTop - return the object on top of the stack as the value of the message \(rightCenter\) Literal frame - 1 `#right` - 2 `#center` - **Receiver:** 50@50 corner: 200@200 - **Arguments:** - **Temporary Variables:** - **Stack:** 200 ### Handling blocks: inlined ones ``` Rectangle >> containsPoint: aPoint + ^ corner x +``` + + + +**Active** Rectangle >> #right +- **==>**<01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack +- <7E> send: x +- <5C> returnTop + +Literal frame +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** + + + +**Suspended** Rectangle >> #rightCenter +- <4C> self - push the receiver \(self\) onto the stack +- <80> send: right - send a unary message with the selector in the first literal \(right\) +- **==>** <4C> self push the receiver \(self\) onto the stack +- <81> send: center - send a unary message with the selector in the first literal \(center\) +- <7F> send: y - send a unary message with the selector in the first literal \(y\) +- <6B> send: @ - send a unary message with the selector in the first literal \(@\) +- <5C> returnTop - return the object on top of the stack as the value of the message \(rightCenter\) + +Literal frame +- 1 `#right` +- 2 `#center` +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** + + + +#### Execution of the right message + + +The next cycle of the interpreter advances the new context instead of the previous one. + +**Active** Rectangle >> #right +- <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack +- **==>**<7E> send: x +- <5C> returnTop + +Literal frame +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** 200@200 + + + +**Suspended** Rectangle >> #rightCenter +- <4C> self - push the receiver \(self\) onto the stack +- <80> send: right - send a unary message with the selector in the first literal \(right\) +- **==>** <4C> self push the receiver \(self\) onto the stack +- <81> send: center - send a unary message with the selector in the first literal \(center\) +- <7F> send: y - send a unary message with the selector in the first literal \(y\) +- <6B> send: @ - send a unary message with the selector in the first literal \(@\) +- <5C> returnTop - return the object on top of the stack as the value of the message \(rightCenter\) + +Literal frame +- 1 `#right` +- 2 `#center` +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** + + +#### Execution of the right message + +In the next cycle, another message is sent, perhaps creating another context. Instead of following the response of this new message \(`x`\), we will skip to the point that this context returns a value \(to `right`\). When the result of `x` has been returned, the new context looks like this: + +**Active** Rectangle >> #right +- <01> pushRcvr: 1 - push the value of the receiver's second instance variable \(corner\) onto the stack +- <7E> send: x +- **==>** <5C> returnTop + +Literal frame +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** 200 + + + +**Suspended** Rectangle >> #rightCenter +- <4C> self - push the receiver \(self\) onto the stack +- <80> send: right - send a unary message with the selector in the first literal \(right\) +- **==>** <4C> self push the receiver \(self\) onto the stack +- <81> send: center - send a unary message with the selector in the first literal \(center\) +- <7F> send: y - send a unary message with the selector in the first literal \(y\) +- <6B> send: @ - send a unary message with the selector in the first literal \(@\) +- <5C> returnTop - return the object on top of the stack as the value of the message \(rightCenter\) + +Literal frame +- 1 `#right` +- 2 `#center` +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** + + + + +#### Returning from right + + +The next bytecode returns the value \(200\) on the top of the active context's stack as the value of the message that created the context \(`right`\). The active context's sender becomes the active context again and the returned value is pushed on its stack. It is ready to continue its execution. + + +**Active** Rectangle >> #rightCenter +- <4C> self - push the receiver \(self\) onto the stack +- <80> send: right - send a unary message with the selector in the first literal \(right\) +- **==>** <4C> self push the receiver \(self\) onto the stack +- <81> send: center - send a unary message with the selector in the first literal \(center\) +- <7F> send: y - send a unary message with the selector in the first literal \(y\) +- <6B> send: @ - send a unary message with the selector in the first literal \(@\) +- <5C> returnTop - return the object on top of the stack as the value of the message \(rightCenter\) + +Literal frame +- 1 `#right` +- 2 `#center` +- **Receiver:** 50@50 corner: 200@200 +- **Arguments:** +- **Temporary Variables:** +- **Stack:** 200 + + + +### Handling blocks: inlined ones + + +``` +Rectangle >> containsPoint: aPoint "Answer whether aPoint is within the receiver." - ^origin <= aPoint and: [aPoint < corner] ``` - <00> pushRcvr: 0 - <40> pushTemp: 0 - <64> send: <= - jumpFalse: 41 - <40> pushTemp: 0 - <01> pushRcvr: 1 - <62> send: < - jumpTo: 42 - 41 <4E> pushConstant: false - 42 <5C> returnTop\)" ### Blocks Context should have a receiver because executing `[ self foo ] value` should execute foo on self and not the block ### Primitive Methods? The interpreter's actions after finding a compiled method depend on whether or not the compiled method indicates that a primitive method may be able to respond to the message. If no primitive method is indicated, a new method context is created and made active as described in previous sections. If a primitive method is indicated in the compiled method, the interpreter may be able to respond to the message without actually executing the bytecodes. For example, one of the primitive methods is associated with the `+` message to instances of `SmallInteger`. ``` SmallInteger >> + addend + ^origin <= aPoint and: [aPoint < corner] +``` + + +- <00> pushRcvr: 0 +- <40> pushTemp: 0 +- <64> send: <= +- jumpFalse: 41 +- <40> pushTemp: 0 +- <01> pushRcvr: 1 +- <62> send: < +- jumpTo: 42 +- 41 <4E> pushConstant: false +- 42 <5C> returnTop\)" + + + +### Blocks + + +Context should have a receiver because executing `[ self foo ] value` should execute foo on self and not the block + + + +### Primitive Methods? + + +The interpreter's actions after finding a compiled method depend on whether or not the compiled method indicates that a primitive method may be able to respond to the message. If no primitive method is indicated, a new method context is created and made active as described in previous sections. If a primitive method is indicated in the compiled method, the interpreter may be able to respond to the message without actually executing the bytecodes. For example, one of the primitive methods is associated with the `+` message to instances of `SmallInteger`. + +``` +SmallInteger >> + addend - ^super + addend ``` - callPrimitive: 1 - <4C> self - push the receiver on the stack - <40> pushTemp: 0 - push the first argument - superSend: `+` - send a message via super - <5C> returnTop - return the object on top of the stack as the value of the message \ No newline at end of file + ^super + addend +``` + + +- callPrimitive: 1 +- <4C> self - push the receiver on the stack +- <40> pushTemp: 0 - push the first argument +- superSend: `+` - send a message via super +- <5C> returnTop - return the object on top of the stack as the value of the message diff --git a/Chapters/BasicsOnExecution/figures/interpreter_activation-ascii.txt b/Chapters/BasicsOnExecution/figures/interpreter_activation-ascii.txt new file mode 100644 index 0000000..ab95539 --- /dev/null +++ b/Chapters/BasicsOnExecution/figures/interpreter_activation-ascii.txt @@ -0,0 +1,7 @@ + +instructionPointer -----> <01> pushRcvr 1 - push the value of the receiver's second instance variable (corner ) onto the stack + <7E> send 'x' - send the unary message x + <00> pushRcvr 0 - push the value of the receiver's first instance variable (origin ) onto the stack + <7E> send 'x' - send the unary message x + <61> send '-' - send the binary message - + <5C> returnTop \ No newline at end of file diff --git a/Chapters/BasicsOnExecution/figures/interpreter_activation.pdf b/Chapters/BasicsOnExecution/figures/interpreter_activation.pdf new file mode 100644 index 0000000000000000000000000000000000000000..12a271bc907fde314e19d857aa447fb36a33f16f GIT binary patch literal 11768 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 pushRcvr: 1 --- push the value of the receiver's second instance variable (corner ) onto the stack -<7E> send: x --- send the unary message x -<00> pushRcvr: 0 --- push the value of the receiver's first instance variable (origin ) onto the stack -<7E> send: x --- send the unary message x -<61> send: - --- send the binary message - -<5C> returnTop - return the object on top of the stack as the value of the message (width ) - + +instructionPtr -----> <01> pushRcvr 1 - push the value of the receiver's second instance variable (corner ) + <7E> send 'x' - send the unary message x + <00> pushRcvr 0 - push the value of the receiver's first instance variable (origin ) + <7E> send 'x' - send the unary message x + <61> send '-' - send the binary message - + <5C> returnTop diff --git a/Chapters/BasicsOnExecution/figures/width_0.pdf b/Chapters/BasicsOnExecution/figures/width_0.pdf index 65d203b988e2e61e342453e96aa6da13e3f78b7e..3db29c0e24dcc1e6004dd7994b721ee7f987edf3 100644 GIT binary patch delta 10240 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 delta 11120 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 + +## Slang +@cha:Slang +Just here so that we can refer to it + + # The Memory Manager