Анимация
JavaScript
|
Главная Библионтека мы продолжим давай сгенерируем синтаксический анализатор достаточный только для обработки одиночного показателя в выражении. В процессе мы также обнаружим, что по необходимости создали также модуль генератора кода. Помните самую первую главу этой серии? Мы считывали целочисленное значение, скажем n, и генерировали код для его загрузки в регистр D0 через move: MOVE #n,D0 Немного погодя, мы повторили этот процесс для переменной, MOVE X(PC),D0 а затем для показателя, который может быть и константой и переменной. В память о прошлом, давайте повторим этот процесс Определите следующий новый модуль: unit Parser; interface uses Input, Scanner, Errors, CodeGen; procedure Factor; implementation { Parse and Translate a Factor } procedure Factor; begin LoadConstant(GetNumber); end; end. Как вы можете видеть, этот модуль вызывает процедуру LoadConstant, которая фактически выполняет вывод ассемблерного кода. Модуль также использует новый модуль CodeGen. Этот шаг представляет последнее главное изменение в нашей архитектуре с более ранних глав: перемещение машино-зависимого кода в отдельный модуль. Если я дойду до конца, вне CodeGen не будет ни одной строчки кода, которая указывала бы на то, что мы нацелены на процессор 68000. И это то место, которое показывает, что моя цель достижима. Для тех из вас, кто желает, чтобы я использовал архитектуру 80x86 (или любую другую) вместо 68000, вот мой ответ: просто замените CodeGen на подходящий для вашего ЦПУ. Пока наш генератор кода содержит только одну процедуру. Вот этот модуль: unit CodeGen; interface uses Output; procedure LoadConstant(n: string); implementation { Load the Primary Register with a Constant } procedure LoadConstant(n: string); begin EmitLnCMOVE # + n + ,D0 ); end; end. Скопируйте и откомпилируйте этот модуль и выполните следующую основную программу: program Main; uses WinCRT, Input, Output, Errors, Scanner, Parser; begin Factor; end. Вот он, сгенерированный код, такой как мы и надеялись. Теперь, я надеюсь, вы можете начать видеть преимущества модульной архитектуры нашего нового проекта. Здесь мы имеем основную программу длиной всего пять строк. Это все, что нам нужно видеть, если мы не захотим видеть больше. И пока все эти модули сидят здесь терпеливо ожидая когда смогут послужить нам. Наше преимущество в том, что мы имеем простой и короткий код, но мощных союзников. Что остается сделать, это расширить модули до уровня возможностей более ранних глав. Мы сделаем это в следующей главе, но прежде, чем я закончу, давайте закончим синтаксический анализ показателя только для того, чтобы убедить себя, что мы знаем как. Конечная версия CodeGen включает новую процедуру LoadVariable: unit CodeGen; interface uses Output; procedure LoadConstant(n: string); procedure LoadVariable(Name: string); implementation { Load the Primary Register with a Constant } procedure LoadConstant(n: string); begin EmitLnCMOVE # + n + ,D0 ); end; { Load a Variable to the Primary Register } procedure LoadVariable(Name: string); begin EmitLnCMOVE + Name + (PC),D0); end; end. Сам модуль Parser не изменяется, но мы имеем более сложную версию процедуры Factor: { Parse and Translate a Factor } procedure Factor; begin if IsDigit(Look) then LoadConstant(GetNumber) else if IsAlpha(Look)then LoadVariable(GetName) else Error(Unrecognized character + Look); end; Теперь, без изменений основной программы, вы должны обнаружить, что программа обрабатывает и переменный и постоянный показатель. К этому моменту наша архитектура почти завершена; у нас есть модули, выполняющие всю грязную работу и достаточно кода в синтаксическом анализаторе и генераторе кода чтобы продемонстрировать что все работает. Остается расширить модули которые мы определили, в особенности синтаксический анализатор и генератор кода, для поддержки более сложных синтаксических элементов, которые составляют настоящий язык. Так как мы делали это много раз прежде в предыдущих главах, не должно занять у нас много времени вернуться назад к тому месту, где мы были до долгого перерыва. Мы продолжим этот процесс в Главе 16, которая скоро появится. Увидимся. ССЫЛКИ 1. Crenshaw, J.W., "Object-Oriented Design of Assemblers and Compilers," Proc. Software Development 91 Conference, Miller Freeman, San Francisco, CA, February 1991, pp. 143-155. 2. Crenshaw, J.W., "A Perfect Marriage," Computer Language, Volume 8, #6, June 1991, pp. 44-55. 3. Crenshaw, J.W., "Syntax-Driven Object-Oriented Design," Proc. 1991 Embedded Systems Conference, Miller Freeman, San Francisco, CA, September 1 991 , pp. 45-60. 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 [ 91 ] 92 93 94 95 96 97 |