Skip to content
geeksforgeeks
  • Tutorials
    • Python
    • Java
    • Data Structures & Algorithms
    • ML & Data Science
    • Interview Corner
    • Programming Languages
    • Web Development
    • CS Subjects
    • DevOps And Linux
    • School Learning
    • Practice Coding Problems
  • Courses
    • DSA to Development
    • Get IBM Certification
    • Newly Launched!
      • Master Django Framework
      • Become AWS Certified
    • For Working Professionals
      • Interview 101: DSA & System Design
      • Data Science Training Program
      • JAVA Backend Development (Live)
      • DevOps Engineering (LIVE)
      • Data Structures & Algorithms in Python
    • For Students
      • Placement Preparation Course
      • Data Science (Live)
      • Data Structure & Algorithm-Self Paced (C++/JAVA)
      • Master Competitive Programming (Live)
      • Full Stack Development with React & Node JS (Live)
    • Full Stack Development
    • Data Science Program
    • All Courses
  • Aptitude
  • Engineering Mathematics
  • Discrete Mathematics
  • Operating System
  • DBMS
  • Computer Networks
  • Digital Logic and Design
  • C Programming
  • Data Structures
  • Algorithms
  • Theory of Computation
  • Compiler Design
  • Computer Org and Architecture
Open In App
Next Article:
Data flow analysis in Compiler
Next article icon

Three address code in Compiler

Last Updated : 27 Dec, 2024
Comments
Improve
Suggest changes
Like Article
Like
Report

TAC is an intermediate representation of three-address code utilized by compilers to ease the process of code generation. Complex expressions are, therefore, decomposed into simple steps comprising, at most, three addresses: two operands and one result using this code. The results from TAC are always stored in the temporary variables that a compiler generates. This design ensures explicit ordering of the operations that come into play. Since it is simple, TAC lends itself nicely to optimization and translation to machine code. TAC represents control flow and data dependencies. It falls somewhere between high-level source code and machine-level instructions. Therefore, it is a very essential phase in compiler designing.

Three Address Codes are Used in Compiler Applications

  • Optimization: Three-address code is often used as an intermediate representation of code during the optimization phases of the compilation process. The three-address code allows the compiler to analyze the code and perform optimizations that can improve the performance of the generated code.
  • Code generation: Three address codes can also be used as an intermediate representation of code during the code generation phase of the compilation process. The three-address code allows the compiler to generate code that is specific to the target platform, while also ensuring that the generated code is correct and efficient.
  • Debugging: Three address codes can be helpful in debugging the code generated by the compiler. Since the address code is a low-level language, it is often easier to read and understand than the final generated code. Developers can use the three address codes to trace the execution of the program and identify errors or issues that may be present.
  • Language translation: Three address codes can also be used to translate code from one programming language to another. By translating code to a common intermediate representation, it becomes easier to translate the code to multiple target languages.

General Representation

a = b
a = op b
a = b op c

Where a, b, or c represents operands like names, constants or compiler-generated temporaries and op represents the operator 

Example-1: Convert the expression a * - (b + c) into three address codes.

Three Address Code

Example 2,: Write three address codes for the following code

for(i = 1; i<=10; i++)
{
a[i] = x * 5;
}
Three address code
3 address code for the for loop

Implementation of Three Address Code

There are 3 representations of three address code namely

  • Quadruple
  • Triples
  • Indirect Triples

1. Quadruple: It is a structure which consists of 4 fields namely op, arg1, arg2 and result. op denotes the operator and arg1 and arg2 denotes the two operands and result is used to store the result of the expression. 

Advantage

  • Easy to rearrange code for global optimization.
  • One can quickly access value of temporary variables using symbol table.

Disadvantage

  • Contain lot of temporaries.
  • Temporary variable creation increases time and space complexity.

Example - Consider expression a = b * - c + b * - c. The three address code is:

t1 = uminus c   (Unary minus operation on c)
t2 = b * t1
t3 = uminus c (Another unary minus operation on c)
t4 = b * t3
t5 = t2 + t4
a = t5 (Assignment of t5 to a)

Quadruple Representation 

2. Triples: This representation doesn't make use of extra temporary variable to represent a single operation instead when a reference to another triple's value is needed, a pointer to that triple is used. So, it consist of only three fields namely op, arg1 and arg2. 

Disadvantage

  • Temporaries are implicit and difficult to rearrange code.
  • It is difficult to optimize because optimization involves moving intermediate code. When a triple is moved, any other triple referring to it must be updated also. With help of pointer one can directly access symbol table entry.

Example - Consider expression a = b * - c + b * - c 

Triples Representation 

3. Indirect Triples This representation makes use of pointer to the listing of all references to computations which is made separately and stored. Its similar in utility as compared to quadruple representation but requires less space than it. Temporaries are implicit and easier to rearrange code. 

Example - Consider expression a = b * - c + b * - c 

Indirect TriplesQuestion - Write quadruple, triples and indirect triples for following expression : (x + y) * (y + z) + (x + y + z)

Explanation - The three address code is:

(1) t1 = x + y
(2) t2 = y + z
(3) t3 = t1 * t2
(4) t4 = t1 + z
(5) t5 = t3 + t4

Quadruple Representation  

Triples Representation   

Indirect Triples

Implementation of Three Address Code

1. Source Code Parsing and Abstract Syntax Tree Generation.

  • Lexical Analysis: source code is converted into tokens, keywords, identifiers, operators, literals, etc.
  • Syntax Analysis: An Abstract Syntax Tree that would be the representation of the grammatical structure of the code.
  • Semantic Analysis: Derivation of semantic errors type mismatches, undeclared variables, etc. and builds a symbol table.

2. TAC Instructions Generation

  • Traversal of the Abstract Syntax Tree: visiting each of the nodes in the abstract syntax tree generates the proper TAC instructions.
  • Use temporaries for intermediate computation.
  • Three address form: Each instruction should have at most three operands, two source and one target.

3. Evaluation of Expressions

  • Arithmetic Expressions: Divide complex expression into simpler expressions by using the temporaries. Example a + b * c which should be encoded as follows:
  • t1 = b * c
  • t2 := a + t1

3. Logical Expressions:

  • Translate the logical AND, OR and NOT operators into conditional jumps. Uses temporary registers.

4. Control Flow Constructs

  • If Statements :Use conditional jumps.
  • While Loops : Involve in unconditional jumps to the loop header and conditional jumps to exit.
  • For Loops: Converted to equivalent while loops.
  • Goto Statements: Unconditionally jump to the specified labels.

5. Procedure Calls

  • Argument Passing : Arguments are pushed onto the stack
  • Function Call: A jump is made to the function's entry point.
  • Return Value: Stored in a temporary variable while popping the arguments off the stack..

Conclusion

TAC is the important intermediate representation major compiler designers use because of its simplicity and efficiency. Complex expressions are broken down into simple, manageable instructions that have a maximum usage of three addresses, therefore making the generation and optimization processes easy. One of the strong points of TAC is flexibility: it handles a large variety of programming constructs while remaining independent of machine architecture. This form also allows optimizations such as constant folding and removal of dead code that make the final executable code considerably leaner. TAC is a useful independent language, situated between high-level source and machine code.

What kind of instructions does TAC comprise?

Other TACs involve arithmetic, AND and OR logical operations, assignment statements, and the control flow instructions of the form of conditional jump or function call. This usually contains two operands with its result usually stored in a temporary or destination variable.

How many operators are usually specified in three-address codes?

The operator set usually includes arithmetic operations-add, subtract; logical operations-AND, OR; comparisons; and control flow-conditional jumps, function calls, load/store for memory access.

In what ways is three-address code different from the other already used formats of intermediate representation?

Three-address code makes both the operation and operands explicit whereas none of the other intermediate representations (ASTs do) are making TAC easier to generate, manipulate, and optimize.


Next Article
Data flow analysis in Compiler

A

Ankit87
Improve
Article Tags :
  • Misc
  • Technical Scripter
  • Compiler Design
  • GATE CS
Practice Tags :
  • Misc

Similar Reads

    Introduction of Compiler Design
    A compiler is software that translates or converts a program written in a high-level language (Source Language) into a low-level language (Machine Language or Assembly Language). Compiler design is the process of developing a compiler.The development of compilers is closely tied to the evolution of
    9 min read

    Compiler Design Basics

    Introduction of Compiler Design
    A compiler is software that translates or converts a program written in a high-level language (Source Language) into a low-level language (Machine Language or Assembly Language). Compiler design is the process of developing a compiler.The development of compilers is closely tied to the evolution of
    9 min read
    Compiler construction tools
    The compiler writer can use some specialized tools that help in implementing various phases of a compiler. These tools assist in the creation of an entire compiler or its parts. Some commonly used compiler construction tools include: Parser Generator - It produces syntax analyzers (parsers) from the
    4 min read
    Phases of a Compiler
    A compiler is a software tool that converts high-level programming code into machine code that a computer can understand and execute. It acts as a bridge between human-readable code and machine-level instructions, enabling efficient program execution. The process of compilation is divided into six p
    10 min read
    Symbol Table in Compiler
    Every compiler uses a symbol table to track all variables, functions, and identifiers in a program. It stores information such as the name, type, scope, and memory location of each identifier. Built during the early stages of compilation, the symbol table supports error checking, scope management, a
    8 min read
    Error Handling in Compiler Design
    During the process of language translation, the compiler can encounter errors. While the compiler might not always know the exact cause of the error, it can detect and analyze the visible problems. The main purpose of error handling is to assist the programmer by pointing out issues in their code. E
    5 min read
    Language Processors: Assembler, Compiler and Interpreter
    Computer programs are generally written in high-level languages (like C++, Python, and Java). A language processor, or language translator, is a computer program that convert source code from one programming language to another language or to machine code (also known as object code). They also find
    5 min read
    Generation of Programming Languages
    Programming languages have evolved significantly over time, moving from fundamental machine-specific code to complex languages that are simpler to write and understand. Each new generation of programming languages has improved, allowing developers to create more efficient, human-readable, and adapta
    6 min read

    Lexical Analysis

    Introduction of Lexical Analysis
    Lexical analysis, also known as scanning is the first phase of a compiler which involves reading the source program character by character from left to right and organizing them into tokens. Tokens are meaningful sequences of characters. There are usually only a small number of tokens for a programm
    6 min read
    Flex (Fast Lexical Analyzer Generator)
    Flex (Fast Lexical Analyzer Generator), or simply Flex, is a tool for generating lexical analyzers scanners or lexers. Written by Vern Paxson in C, circa 1987, Flex is designed to produce lexical analyzers that is faster than the original Lex program. Today it is often used along with Berkeley Yacc
    7 min read
    Introduction of Finite Automata
    Finite automata are abstract machines used to recognize patterns in input sequences, forming the basis for understanding regular languages in computer science. They consist of states, transitions, and input symbols, processing each symbol step-by-step. If the machine ends in an accepting state after
    4 min read
    Classification of Context Free Grammars
    A Context-Free Grammar (CFG) is a formal rule system used to describe the syntax of programming languages in compiler design. It provides a set of production rules that specify how symbols (terminals and non-terminals) can be combined to form valid sentences in the language. CFGs are important in th
    4 min read
    Ambiguous Grammar
    Context-Free Grammars (CFGs) is a way to describe the structure of a language, such as the rules for building sentences in a language or programming code. These rules help define how different symbols can be combined to create valid strings (sequences of symbols).CFGs can be divided into two types b
    7 min read

    Syntax Analysis & Parsers

    Introduction to Syntax Analysis in Compiler Design
    Syntax Analysis (also known as parsing) is the step after Lexical Analysis. The Lexical analysis breaks source code into tokens.Tokens are inputs for Syntax Analysis.The goal of Syntax Analysis is to interpret the meaning of these tokens. It checks whether the tokens produced by the lexical analyzer
    7 min read
    FIRST and FOLLOW in Compiler Design
    In compiler design, FIRST and FOLLOW are two sets used to help parsers understand how to process a grammar.FIRST Set: The FIRST set of a non-terminal contains all the terminal symbols that can appear at the beginning of any string derived from that non-terminal. In other words, it tells us which ter
    6 min read
    Parsing - Introduction to Parsers
    Parsing, also known as syntactic analysis, is the process of analyzing a sequence of tokens to determine the grammatical structure of a program. It takes the stream of tokens, which are generated by a lexical analyzer or tokenizer, and organizes them into a parse tree or syntax tree.The parse tree v
    6 min read
    Construction of LL(1) Parsing Table
    Parsing is an essential part of computer science, especially in compilers and interpreters. From the various parsing techniques, LL(1) parsing is best. It uses a predictive, top-down approach. This allows efficient parsing without backtracking. This article will explore parsing and LL(1) parsing. It
    6 min read

    Syntax Directed Translation & Intermediate Code Generation

    Syntax Directed Translation in Compiler Design
    Syntax-Directed Translation (SDT) is a method used in compiler design to convert source code into another form while analyzing its structure. It integrates syntax analysis (parsing) with semantic rules to produce intermediate code, machine code, or optimized instructions.In SDT, each grammar rule is
    8 min read
    S - Attributed and L - Attributed SDTs in Syntax Directed Translation
    In Syntax-Directed Translation (SDT), the rules are those that are used to describe how the semantic information flows from one node to the other during the parsing phase. SDTs are derived from context-free grammars where referring semantic actions are connected to grammar productions. Such action c
    4 min read
    Parse Tree and Syntax Tree
    Parse Tree and Syntax tree are tree structures that represent the structure of a given input according to a formal grammar. They play an important role in understanding and verifying whether an input string aligns with the language defined by a grammar. These terms are often used interchangeably but
    4 min read
    Intermediate Code Generation in Compiler Design
    In the analysis-synthesis model of a compiler, the front end of a compiler translates a source program into an independent intermediate code, then the back end of the compiler uses this intermediate code to generate the target code (which can be understood by the machine). The benefits of using mach
    6 min read
    Issues in the design of a code generator
    A code generator is a crucial part of a compiler that converts the intermediate representation of source code into machine-readable instructions. Its main task is to produce the correct and efficient code that can be executed by a computer. The design of the code generator should ensure that it is e
    7 min read
    Three address code in Compiler
    TAC is an intermediate representation of three-address code utilized by compilers to ease the process of code generation. Complex expressions are, therefore, decomposed into simple steps comprising, at most, three addresses: two operands and one result using this code. The results from TAC are alway
    6 min read
    Data flow analysis in Compiler
    Data flow is analysis that determines the information regarding the definition and use of data in program. With the help of this analysis, optimization can be done. In general, its process in which values are computed using data flow analysis. The data flow property represents information that can b
    6 min read

    Code Optimization & Runtime Environments

    Code Optimization in Compiler Design
    Code optimization is a crucial phase in compiler design aimed at enhancing the performance and efficiency of the executable code. By improving the quality of the generated machine code optimizations can reduce execution time, minimize resource usage, and improve overall system performance. This proc
    9 min read
    Introduction of Object Code in Compiler Design
    Let assume that you have a C program then, you give it to the compiler and compiler will produce the output in assembly code. Now, that assembly language code will be given to the assembler and assembler will produce some code and that code is known as Object Code. Object CodeObject Code is a key co
    6 min read
    Static and Dynamic Scoping
    The scope of a variable x in the region of the program in which the use of x refers to its declaration. One of the basic reasons for scoping is to keep variables in different parts of the program distinct from one another. Since there are only a small number of short variable names, and programmers
    6 min read
    Runtime Environments in Compiler Design
    A translation needs to relate the static source text of a program to the dynamic actions that must occur at runtime to implement the program. The program consists of names for procedures, identifiers, etc., that require mapping with the actual memory location at runtime. Runtime environment is a sta
    8 min read
    Linker
    A linker is an essential tool in the process of compiling a program. It helps combine various object modules (output from the assembler) into a single executable file that can be run on a system. The linker’s job is to manage and connect different pieces of code and data, ensuring that all reference
    8 min read
    Loader in C/C++
    The loader is the program of the operating system which loads the executable from the disk into the primary memory(RAM) for execution. It allocates the memory space to the executable module in the main memory and then transfers control to the beginning instruction of the program. The loader is an im
    3 min read

    Practice Questions

    Last Minute Notes - Compiler Design
    In computer science, compiler design is the study of how to build a compiler, which is a program that translates high-level programming languages (like Python, C++, or Java) into machine code that a computer's hardware can execute directly. The focus is on how the translation happens, ensuring corre
    13 min read
    Compiler Design - GATE CSE Previous Year Questions
    In this article, we are mainly focusing on the Compiler Design GATE Questions that have been asked in Previous Years, with their solutions. And where an explanation is required, we have also provided the reason. Topic-Wise Quizzes to Practice Previous Year's QuestionsLexical AnalysisParsingSyntax-Di
    1 min read
geeksforgeeks-footer-logo
Corporate & Communications Address:
A-143, 7th Floor, Sovereign Corporate Tower, Sector- 136, Noida, Uttar Pradesh (201305)
Registered Address:
K 061, Tower K, Gulshan Vivante Apartment, Sector 137, Noida, Gautam Buddh Nagar, Uttar Pradesh, 201305
GFG App on Play Store GFG App on App Store
Advertise with us
  • Company
  • About Us
  • Legal
  • Privacy Policy
  • In Media
  • Contact Us
  • Advertise with us
  • GFG Corporate Solution
  • Placement Training Program
  • Languages
  • Python
  • Java
  • C++
  • PHP
  • GoLang
  • SQL
  • R Language
  • Android Tutorial
  • Tutorials Archive
  • DSA
  • Data Structures
  • Algorithms
  • DSA for Beginners
  • Basic DSA Problems
  • DSA Roadmap
  • Top 100 DSA Interview Problems
  • DSA Roadmap by Sandeep Jain
  • All Cheat Sheets
  • Data Science & ML
  • Data Science With Python
  • Data Science For Beginner
  • Machine Learning
  • ML Maths
  • Data Visualisation
  • Pandas
  • NumPy
  • NLP
  • Deep Learning
  • Web Technologies
  • HTML
  • CSS
  • JavaScript
  • TypeScript
  • ReactJS
  • NextJS
  • Bootstrap
  • Web Design
  • Python Tutorial
  • Python Programming Examples
  • Python Projects
  • Python Tkinter
  • Python Web Scraping
  • OpenCV Tutorial
  • Python Interview Question
  • Django
  • Computer Science
  • Operating Systems
  • Computer Network
  • Database Management System
  • Software Engineering
  • Digital Logic Design
  • Engineering Maths
  • Software Development
  • Software Testing
  • DevOps
  • Git
  • Linux
  • AWS
  • Docker
  • Kubernetes
  • Azure
  • GCP
  • DevOps Roadmap
  • System Design
  • High Level Design
  • Low Level Design
  • UML Diagrams
  • Interview Guide
  • Design Patterns
  • OOAD
  • System Design Bootcamp
  • Interview Questions
  • Inteview Preparation
  • Competitive Programming
  • Top DS or Algo for CP
  • Company-Wise Recruitment Process
  • Company-Wise Preparation
  • Aptitude Preparation
  • Puzzles
  • School Subjects
  • Mathematics
  • Physics
  • Chemistry
  • Biology
  • Social Science
  • English Grammar
  • Commerce
  • World GK
  • GeeksforGeeks Videos
  • DSA
  • Python
  • Java
  • C++
  • Web Development
  • Data Science
  • CS Subjects
@GeeksforGeeks, Sanchhaya Education Private Limited, All rights reserved
We use cookies to ensure you have the best browsing experience on our website. By using our site, you acknowledge that you have read and understood our Cookie Policy & Privacy Policy
Lightbox
Improvement
Suggest Changes
Help us improve. Share your suggestions to enhance the article. Contribute your expertise and make a difference in the GeeksforGeeks portal.
geeksforgeeks-suggest-icon
Create Improvement
Enhance the article with your expertise. Contribute to the GeeksforGeeks community and help create better learning resources for all.
geeksforgeeks-improvement-icon
Suggest Changes
min 4 words, max Words Limit:1000

Thank You!

Your suggestions are valuable to us.

What kind of Experience do you want to share?

Interview Experiences
Admission Experiences
Career Journeys
Work Experiences
Campus Experiences
Competitive Exam Experiences