Control-Flow Bending: on the Effectiveness of Control-Flow Integrity

Total Page:16

File Type:pdf, Size:1020Kb

Control-Flow Bending: on the Effectiveness of Control-Flow Integrity Control-Flow Bending: On the Effectiveness of Control-Flow Integrity Nicolas Carlini, University of California, Berkeley; Antonio Barresi, ETH Zürich; Mathias Payer, Purdue University; David Wagner, University of California, Berkeley; Thomas R. Gross, ETH Zürich https://www.usenix.org/conference/usenixsecurity15/technical-sessions/presentation/carlini This paper is included in the Proceedings of the 24th USENIX Security Symposium August 12–14, 2015 • Washington, D.C. ISBN 978-1-939133-11-3 Open access to the Proceedings of the 24th USENIX Security Symposium is sponsored by USENIX Control-Flow Bending: On the Effectiveness of Control-Flow Integrity Nicolas Carlini Antonio Barresi Mathias Payer David Wagner UC Berkeley ETH Zurich Purdue University UC Berkeley Thomas R. Gross ETH Zurich Abstract Defenses that prevent the corruption stop the actual Control-Flow Integrity (CFI) is a defense which pre- memory corruption before it can do any harm to the pro- vents control-flow hijacking attacks. While recent re- gram (i.e., no attacker-controlled values are ever used search has shown that coarse-grained CFI does not stop out-of-context). Examples for prevent-the-corruption attacks, fine-grained CFI is believed to be secure. defenses are SoftBound [22], Data-Flow Integrity [6], We argue that assessing the effectiveness of practi- or Code-Pointer Integrity [18]. In contrast, prevent- cal CFI implementations is non-trivial and that common the-exploit defenses allow memory corruption to occur evaluation metrics fail to do so. We then evaluate fully- but protect the application from subsequent exploitation; precise static CFI — the most restrictive CFI policy that they try to survive or tolerate adversarial corruption of does not break functionality — and reveal limitations in memory. Examples for prevent-the-exploit defenses are its security. Using a generalization of non-control-data DEP [38] or stack canaries [10]. attacks which we call Control-Flow Bending (CFB), we Control-Flow Integrity (CFI) [1, 3, 12, 15, 27, 30, 31, show how an attacker can leverage a memory corruption 39, 41–44] is a promising stateless prevent-the-exploit vulnerability to achieve Turing-complete computation on defense mechanism that aims for complete protection memory using just calls to the standard library. We use against control-flow hijacking attacks under a threat this attack technique to evaluate fully-precise static CFI model with a powerful attacker that can read and write on six real binaries and show that in five out of six cases, into the process’s address space. CFI ensures that pro- powerful attacks are still possible. Our results suggest gram execution follows a valid path through the static that CFI may not be a reliable defense against memory Control-Flow Graph (CFG). Any deviation from the corruption vulnerabilities. CFG is a CFI violation, terminating the application. CFI We further evaluate shadow stacks in combination is not specific to any particular exploitation vector for with CFI and find that their presence for security is nec- control-flow hijacking. Rather, it enforces its policy on essary: deploying shadow stacks removes arbitrary code all indirect branch instructions. Therefore any attempt by execution capabilities of attackers in three of six cases. an attacker to alter the control-flow in an invalid manner will be detected, regardless of how the attacker changes 1 Introduction the target of the control-flow transfer instruction. CFI is often coupled with a protected shadow stack, Attacking software systems by exploiting memory- which ensures that each return statement matches the corruption vulnerabilities is one of the most common corresponding call and thereby prevents an attacker from attack methods today according to the list of Common tampering with return addresses. While the foundational Vulnerabilities and Exposures. To counter these threats, work [1, 15] included a shadow stack as part of CFI, several hardening techniques have been widely adopted, some more recent research has explored variants of CFI including ASLR [29], DEP [38], and stack canaries [10]. that omit the shadow stack for better performance [9]. Each has limitations: stack canaries protect only against Whereas conformance to the CFG is a stateless policy, contiguous overwrites of the stack, DEP protects against shadow stacks are inherently dynamic and are more pre- code injection but not against code reuse, and ASLR does cise than any static policy can be with respect to returns. not protect against information leakage. Many prior attacks on CFI have focused on attacking a We classify defense mechanisms into two broad cat- weak or suboptimal implementation of CFI. Our focus is egories: prevent-the-corruption and prevent-the-exploit. on evaluating the effectiveness of CFI in its best achiev- USENIX Association 24th USENIX Security Symposium 161 able form, instead of artifacts of some (possibly weak) 2 Background and software attacks CFI implementation. We define fully-precise static CFI as the best achievable CFI policy as follows: a branch Over the past few decades, one of the most common at- from one instruction to another is allowed if and only tack vectors has been exploitation of memory corruption if some benign execution makes that same control-flow within programs written in memory-unsafe languages.In transfer. Such a policy could be imagined as taking any response, operating systems and compilers have started CFG over-approximation and removing edges until re- to support countermeasures against specific exploitation moving additional edges would break functionality. vectors and vulnerability types, but current hardening Thus, fully-precise static CFI is the most restrictive techniques are still unable to stop all attacks. We briefly stateless CFI policy that still allows the program to run as provide an overview of these attacks; more information intended. Both coarse-grained and fine-grained CFI are may be found elsewhere [37]. less precise than fully-precise static CFI, because they both over-approximate the set of valid targets for each 2.1 Control-Flow Hijacking indirect transfer (though to a different degree). In con- trast, fully-precise static CFI involves no approximation One way to exploit a memory corruption bug involves by definition. We acknowledge that fully-precise static hijacking control flow to execute attacker-supplied or CFI might be stricter than anything that can be prac- already-existing code in an application’s address space. tically implemented, but this makes any attacks all the These methods leverage the memory corruption bug to more meaningful: our results help us understand funda- change the target of an indirect branch instruction (ret, mental limits on the effectiveness of the strongest possi- jmp *, or call *). By doing so, an attacker can completely ble CFI policy. control the next instructions to execute. Through several methods of evaluation, we argue that fully-precise static CFI is neither completely broken (as 2.2 Code-Reuse Attacks most coarse-grained defenses are) nor totally secure. We explore what CFI can and cannot prevent, and hope that Data Execution Prevention (DEP) prevents executing this will stimulate a broader discussion about ways to fur- attacker-injected code. However, redirecting control- ther strengthen CFI. flow to already-existing executable code in memory re- We evaluate the security of fully-precise static CFI mains feasible. One technique, return-to-libc [25, 36], both with and without shadow stacks. Recent research reuses existing functions in the address space of the vul- achieves better performance by omitting the shadow nerable process. Runtime libraries (such as libc) often stack in favor of a static policy on return statements. We provide powerful functions, e.g., wrapper functions for still call it fully-precise static CFI when we have added most system calls. One example is libc’s system() a shadow stack, because the shadow stack is orthogonal. function, which allows the attacker to execute shell com- This does not change the fact that the CFI policy is static. mands. Code-reuse attacks are possible when attacker- CFI works by preventing an attacker from deviating needed code is already available in the address space of from the control-flow graph. Our attacks do not involve a vulnerable process. breaking the CFI mechanism itself: we even assume the mechanism is implemented perfectly to its fullest extent. 2.3 Return Oriented Programming Rather, our analysis demonstrates that an attacker can still create exploits for most real applications, without Return Oriented Programming (ROP) [25, 36] is a more causing execution to deviate from the control-flow graph. advanced form of code-reuse attack that lets the attacker This paper provides the following contributions: perform arbitrary computation solely by reusing existing code. It relies upon short instruction sequences (called 1. formalization and evaluation of a space of different “gadgets”) that end with an indirect branch instruction. kinds of CFI schemes; This allows them to be chained, so the attacker can 2. new attacks on fully-precise static CFI, which reveal perform arbitrary computation by executing a carefully- fundamental limits on the effectiveness of CFI; chosen sequence of gadgets. ROP can be generalized 3. evidence that existing metrics for CFI security are to use indirect jump or call instructions instead of re- ineffective; turns [4, 7]. 4. evidence that CFI without a shadow stack is broken; 5. widely applicable Turing-complete attacks on CFI with shadow stacks; and, 2.4 Non-Control-Data Attacks 6. practical case studies of the security of fully-precise A non-control-data attack [8] is an attack where a mem- static CFI for several existing applications. ory corruption vulnerability is used to corrupt only data, 2 162 24th USENIX Security Symposium USENIX Association but not any code pointer. (A code pointer is a pointer 3 Threat model and attacker goals which refers to the code segment, for example, a re- turn address or function pointer.) Depending on the Threat model.
Recommended publications
  • 7. Control Flow First?
    Copyright (C) R.A. van Engelen, FSU Department of Computer Science, 2000-2004 Ordering Program Execution: What is Done 7. Control Flow First? Overview Categories for specifying ordering in programming languages: Expressions 1. Sequencing: the execution of statements and evaluation of Evaluation order expressions is usually in the order in which they appear in a Assignments program text Structured and unstructured flow constructs 2. Selection (or alternation): a run-time condition determines the Goto's choice among two or more statements or expressions Sequencing 3. Iteration: a statement is repeated a number of times or until a Selection run-time condition is met Iteration and iterators 4. Procedural abstraction: subroutines encapsulate collections of Recursion statements and subroutine calls can be treated as single Nondeterminacy statements 5. Recursion: subroutines which call themselves directly or indirectly to solve a problem, where the problem is typically defined in terms of simpler versions of itself 6. Concurrency: two or more program fragments executed in parallel, either on separate processors or interleaved on a single processor Note: Study Chapter 6 of the textbook except Section 7. Nondeterminacy: the execution order among alternative 6.6.2. constructs is deliberately left unspecified, indicating that any alternative will lead to a correct result Expression Syntax Expression Evaluation Ordering: Precedence An expression consists of and Associativity An atomic object, e.g. number or variable The use of infix, prefix, and postfix notation leads to ambiguity An operator applied to a collection of operands (or as to what is an operand of what arguments) which are expressions Fortran example: a+b*c**d**e/f Common syntactic forms for operators: The choice among alternative evaluation orders depends on Function call notation, e.g.
    [Show full text]
  • A Survey of Hardware-Based Control Flow Integrity (CFI)
    A survey of Hardware-based Control Flow Integrity (CFI) RUAN DE CLERCQ∗ and INGRID VERBAUWHEDE, KU Leuven Control Flow Integrity (CFI) is a computer security technique that detects runtime attacks by monitoring a program’s branching behavior. This work presents a detailed analysis of the security policies enforced by 21 recent hardware-based CFI architectures. The goal is to evaluate the security, limitations, hardware cost, performance, and practicality of using these policies. We show that many architectures are not suitable for widespread adoption, since they have practical issues, such as relying on accurate control flow model (which is difficult to obtain) or they implement policies which provide only limited security. CCS Concepts: • Security and privacy → Hardware-based security protocols; Information flow control; • General and reference → Surveys and overviews; Additional Key Words and Phrases: control-flow integrity, control-flow hijacking, return oriented programming, shadow stack ACM Reference format: Ruan de Clercq and Ingrid Verbauwhede. YYYY. A survey of Hardware-based Control Flow Integrity (CFI). ACM Comput. Surv. V, N, Article A (January YYYY), 27 pages. https://doi.org/10.1145/nnnnnnn.nnnnnnn 1 INTRODUCTION Today, a lot of software is written in memory unsafe languages, such as C and C++, which introduces memory corruption bugs. This makes software vulnerable to attack, since attackers exploit these bugs to make the software misbehave. Modern Operating Systems (OSs) and microprocessors are equipped with security mechanisms to protect against some classes of attacks. However, these mechanisms cannot defend against all attack classes. In particular, Code Reuse Attacks (CRAs), which re-uses pre-existing software for malicious purposes, is an important threat that is difficult to protect against.
    [Show full text]
  • Control-Flow Analysis of Functional Programs
    Control-flow analysis of functional programs JAN MIDTGAARD Department of Computer Science, Aarhus University We present a survey of control-flow analysis of functional programs, which has been the subject of extensive investigation throughout the past 30 years. Analyses of the control flow of functional programs have been formulated in multiple settings and have led to many different approximations, starting with the seminal works of Jones, Shivers, and Sestoft. In this paper, we survey control-flow analysis of functional programs by structuring the multitude of formulations and approximations and comparing them. Categories and Subject Descriptors: D.3.2 [Programming Languages]: Language Classifica- tions—Applicative languages; F.3.1 [Logics and Meanings of Programs]: Specifying and Ver- ifying and Reasoning about Programs General Terms: Languages, Theory, Verification Additional Key Words and Phrases: Control-flow analysis, higher-order functions 1. INTRODUCTION Since the introduction of high-level languages and compilers, much work has been devoted to approximating, at compile time, which values the variables of a given program may denote at run time. The problem has been named data-flow analysis or just flow analysis. In a language without higher-order functions, the operator of a function call is apparent from the text of the program: it is a lexically visible identifier and therefore the called function is available at compile time. One can thus base an analysis for such a language on the textual structure of the program, since it determines the exact control flow of the program, e.g., as a flow chart. On the other hand, in a language with higher-order functions, the operator of a function call may not be apparent from the text of the program: it can be the result of a computation and therefore the called function may not be available until run time.
    [Show full text]
  • Obfuscating C++ Programs Via Control Flow Flattening
    Annales Univ. Sci. Budapest., Sect. Comp. 30 (2009) 3-19 OBFUSCATING C++ PROGRAMS VIA CONTROL FLOW FLATTENING T. L¶aszl¶oand A.¶ Kiss (Szeged, Hungary) Abstract. Protecting a software from unauthorized access is an ever de- manding task. Thus, in this paper, we focus on the protection of source code by means of obfuscation and discuss the adaptation of a control flow transformation technique called control flow flattening to the C++ lan- guage. In addition to the problems of adaptation and the solutions pro- posed for them, a formal algorithm of the technique is given as well. A prototype implementation of the algorithm presents that the complexity of a program can show an increase as high as 5-fold due to the obfuscation. 1. Introduction Protecting a software from unauthorized access is an ever demanding task. Unfortunately, it is impossible to guarantee complete safety, since with enough time given, there is no unbreakable code. Thus, the goal is usually to make the job of the attacker as di±cult as possible. Systems can be protected at several levels, e.g., hardware, operating system or source code. In this paper, we focus on the protection of source code by means of obfuscation. Several code obfuscation techniques exist. Their common feature is that they change programs to make their comprehension di±cult, while keep- ing their original behaviour. The simplest technique is layout transformation [1], which scrambles identi¯ers in the code, removes comments and debug informa- tion. Another technique is data obfuscation [2], which changes data structures, 4 T. L¶aszl¶oand A.¶ Kiss e.g., by changing variable visibilities or by reordering and restructuring arrays.
    [Show full text]
  • Control Flow Statements
    Control Flow Statements Christopher M. Harden Contents 1 Some more types 2 1.1 Undefined and null . .2 1.2 Booleans . .2 1.2.1 Creating boolean values . .3 1.2.2 Combining boolean values . .4 2 Conditional statements 5 2.1 if statement . .5 2.1.1 Using blocks . .5 2.2 else statement . .6 2.3 Tertiary operator . .7 2.4 switch statement . .8 3 Looping constructs 10 3.1 while loop . 10 3.2 do while loop . 11 3.3 for loop . 11 3.4 Further loop control . 12 4 Try it yourself 13 1 1 Some more types 1.1 Undefined and null The undefined type has only one value, undefined. Similarly, the null type has only one value, null. Since both types have only one value, there are no operators on these types. These types exist to represent the absence of data, and their difference is only in intent. • undefined represents data that is accidentally missing. • null represents data that is intentionally missing. In general, null is to be used over undefined in your scripts. undefined is given to you by the JavaScript interpreter in certain situations, and it is useful to be able to notice these situations when they appear. Listing 1 shows the difference between the two. Listing 1: Undefined and Null 1 var name; 2 // Will say"Hello undefined" 3 a l e r t( "Hello" + name); 4 5 name= prompt( "Do not answer this question" ); 6 // Will say"Hello null" 7 a l e r t( "Hello" + name); 1.2 Booleans The Boolean type has two values, true and false.
    [Show full text]
  • Control Flow Statements in Java Tutorials Point
    Control Flow Statements In Java Tutorials Point Cnidarian and Waldenses Bubba pillar inclemently and excorticated his mong troublesomely and hereabout. andRounding convective and conversational when energises Jodi some cudgelled Anderson some very prokaryote intertwiningly so wearily! and pardi? Is Gordon always scaphocephalous Go a certain section, commercial use will likely somewhere in java, but is like expression representation of flow control for loop conditionals are independent prognostic factors In this disease let's delve deep into Java Servlets and understand when this. Java Control Flow Statements CoreJavaGuru. Loops in Java Tutorialspointdev. Advantages of Java programming Language. Matlab object On what Kitchen Floor. The flow of controls cursor positioning of files correctly for the python and we represent the location of our java entry points and times as when exploits are handled by minimizing the. The basic control flow standpoint the typecase construct one be seen to realize similar to. Example- Circumference of Circle 227 x Diameter Here This technique. GPGPU GPU Java JCuda For example charity is not discount to control utiliza- com A. There are 3 types of two flow statements supported by the Java programming language Decision-making statements if-then they-then-else switch Looping. Java IfElse Tutorial W3Schools. Spring batch passing data between steps. The tutorial in? Remove that will need to point handling the tutorial will be used or run unit of discipline when using this statement, easing the blinking effect. Either expressed or database systems support for tcl as a controlling source listing applies to a sending field identifier names being part and production.
    [Show full text]
  • Control Flow Analysis
    SIGPLAN Notices 1970 July Control Flow Analysis Frances E. Allen IBM CORPORATION INTRODUCTION Any static, global analysis of the expression and data relation- ships in a program requires a knowledge of the control flow of the program. Since one of the primary reasons for doing such a global analysis in a compiler is to produce optimized programs, control flow analysis has been embedded in many compilers and has been described in several papers. An early paper by Prosser [5] described the use of Boolean matrices (or, more particularly, connectivity matrices) in flow analysis. The use of "dominance" relationships in flow analysis was first introduced by Prosser and much expanded by Lowry and Medlock [6]. References [6,8,9] describe compilers which use various forms of control flow analysis for optimization. Some recent develop- ments in the area are reported in [4] and in [7]. The underlying motivation in all the different types of control flow analysis is the need to codify the flow relationships in the program. The codification may be in connectivity matrices, in predecessor-successor tables, in dominance lists, etc. Whatever the form, the purpose is to facilitate determining what the flow relation- ships are; in other words to facilitate answering such questions as: is this an inner loop?, if an expression is removed from the loop where can it be correctly and profitably placed?, which variable definitions can affect this use? In this paper the basic control flow relationships are expressed in a directed graph. Various graph constructs are then found and shown to codify interesting global relationships.
    [Show full text]
  • Code Based Analysis of Object-Oriented Systems Using Extended Control Flow Graph
    CORE Metadata, citation and similar papers at core.ac.uk Provided by ethesis@nitr Code Based Analysis of Object-Oriented Systems Using Extended Control Flow Graph A thesis submitted in partial fulfillment for the degree of Bachelor of Technology by Shariq Islam 109CS0324 Under the supervision of Prof. S. K. Rath DEPARTMENT OF COMPUTER SCIENCE & ENGINEERING NATIONAL INSTITUTE OF TECHNOLOGY, ROURKELA, ORISSA, INDIA-769008 May 2013 Certificate This is to certify that the thesis entitled \Code Based Analysis of Object-Oriented Systems using Extended Control Flow Graph" submitted by Shariq Islam, in the partial fulfillment of the requirements for the award of Bachelor of Technology Degree in Computer Science & Engineering at National Institute of Technology Rourkela is an authentic work carried out by him under my supervision and guidance. To best of my knowledge, The matter embodied in the thesis has not been submitted to any other university / institution for the award of any Degree or Diploma. Date: Dr. S. K. Rath ||||||||||||||||||||{ Acknowledgements I am grateful to numerous peers who have contributed toward shaping this project. At the outset, I would like to express my sincere thanks to Prof. S. K. Rath for his advice during my project work. As my supervisor, he has constantly encouraged me to remain focused on achieving my goal. His observation and comments helped me to move forward with investigation depth. He has helped me greatly and always been a source of knowledge. I am thankful to all my friends. I sincerely thank everyone who has provided me with inspirational words, a welcome ear, new ideas, constructive criticism, and their invaluable time.
    [Show full text]
  • Computer Science 2. Conditionals and Loops
    COMPUTER SCIENCE COMPUTER SCIENCE SEDGEWICK/WAYNE SEDGEWICK/WAYNE PART I: PROGRAMMING IN JAVA PART I: PROGRAMMING IN JAVA Computer Science 2. Conditionals & Loops •Conditionals: the if statement Computer 2. Conditionals and loops •Loops: the while statement Science •An alternative: the for loop An Interdisciplinary Approach •Nesting R O B E R T S E D G E W I C K 1.3 KEVIN WAYNE •Debugging http://introcs.cs.princeton.edu CS.2.A.Loops.If Context: basic building blocks for programming Conditionals and Loops Control flow • The sequence of statements that are actually executed in a program. any program you might want to write • Conditionals and loops enable us to choreograph control flow. objects true boolean 1 functions and modules statement 1 graphics, sound, and image I/O false statement 2 statement 1 arrays boolean 2 true statement 2 conditionalsconditionals and loops statement 3 Math text I/O false statement 4 statement 3 primitive data types assignment statements This lecture: to infinity and beyond! Previous lecture: straight-line control flow control flow with conditionals and a loop equivalent to a calculator [ previous lecture ] [this lecture] 3 4 The if statement Example of if statement use: simulate a coin flip Execute certain statements depending on the values of certain variables. • Evaluate a boolean expression. public class Flip • If true, execute a statement. { • The else option: If false, execute a different statement. public static void main(String[] args) { if (Math.random() < 0.5) System.out.println("Heads"); Example: if (x > y) max = x; Example: if (x < 0) x = -x; else else max = y; System.out.println("Tails"); % java Flip } Heads } true true x < 0 ? false x > y ? false % java Flip Heads x = -x; max = x; max = y; % java Flip Tails % java Flip Heads Replaces x with the absolute value of x Computes the maximum of x and y 5 6 Example of if statement use: 2-sort Pop quiz on if statements Q.
    [Show full text]
  • Statement-Level Control Structures
    Chapter 7 Levels of Control Flow: 1. Within expressions 2. Among program units 3. Among program statements Evolution: - FORTRAN I control statements were based directly on IBM 704 hardware - Much research and argument in the1960s about the issue - One important result: It was proven that all flowcharts can be coded with only two-way selection and pretest logical loops Def: A control structure is a control statement and the statements whose execution it controls Overall Design Question: What control statements should a language have, beyond selection and pretest logical loops? Copyright © 1998 by Addison Wesley Longman, Inc. 1 Chapter 7 Compound statements - introduced by ALGOL 60 in the form of begin...end A block is a compound statement that can define a new scope (with local variables) Selection Statements Design Issues: 1. What is the form and type of the control expression? 2. What is the selectable segment form (single statement, statement sequence, compound statement)? 3. How should the meaning of nested selectors be specified? Single-Way Examples FORTRAN IF: IF (boolean_expr) statement Problem: can select only a single statement; to select more, a goto must be used, as in the following example Copyright © 1998 by Addison Wesley Longman, Inc. 2 Chapter 7 FORTRAN example: IF (.NOT. condition) GOTO 20 ... ... 20 CONTINUE ALGOL 60 if: if (boolean_expr) then begin ... end Two-way Selector Examples ALGOL 60 if: if (boolean_expr) then statement (the then clause) else statement (the else clause) - The statements could be single or compound Copyright © 1998 by Addison Wesley Longman, Inc. 3 Chapter 7 Nested Selectors e.g.
    [Show full text]
  • Stimulus Structures and Mental Representations in Expert Comprehension of Computer Programs By: Nancy Pennington Presented By: Michael John Decker Author & Journal
    Stimulus Structures and Mental Representations in Expert Comprehension of Computer Programs By: Nancy Pennington Presented by: Michael John Decker Author & Journal • Nancy Pennington - No internet presence • Journal: Cognitive Psychology - Is. 19, Pg: 295-341 (1987) • 2013 JIF: 3.571 • Paper Citations: 525 Research Question • What mental model best describe how an expert programmer builds up knowledge of a program? • When studying the code? • When performing maintenance activities? Motivation • Study of program comprehension means studying role of particular kinds of knowledge in cognitive skill domains • Estimated 50% of professional programmers time is spent on program maintenance • comprehension is a significant part of maintenance • Increased understanding of how knowledge is obtained and applied then use for higher productivity and decreased maintenance costs Computer Program Text Abstractions • Goal hierarchy - functional achievements of program • Data flow - transformations applied to data objects • Control flow - sequences of program actions and the passage of control between them • Conditionalized Actions - representation specifying states of the program and the actions invoked Programming Knowledge • Text Structure Knowledge • Plan Knowledge Text Structure Knowledge • Decomposition of a program’s text into text structure units • Segmentation of statements into phrase-like grouping that are combined into higher-order groupings • Syntactic markings provide clues into the boundary of segments • Segmentation reflects the control structure
    [Show full text]
  • Control Flow
    Control Flow Stephen A. Edwards Columbia University Fall 2012 Control Flow “Time is Nature’s way of preventing everything from happening at once.” Scott identifies seven manifestations of this: 1. Sequencing foo(); bar(); 2. Selection if (a) foo(); 3. Iteration while (i<10) foo(i); 4. Procedures foo(10,20); 5. Recursion foo(int i) { foo(i-1); } 6. Concurrency foo() bar() jj 7. Nondeterminism do a -> foo(); [] b -> bar(); Ordering Within Expressions What code does a compiler generate for a = b + c + d; Most likely something like tmp = b + c; a = tmp + d; (Assumes left-to-right evaluation of expressions.) Order of Evaluation Why would you care? Expression evaluation can have side-effects. Floating-point numbers don’t behave like numbers. GCC returned 25. Sun’s C compiler returned 20. C says expression evaluation order is implementation-dependent. Side-effects int x = 0; int foo() { x += 5; return x; } int bar() { int a = foo() + x + foo(); return a; } What does bar() return? Side-effects int x = 0; int foo() { x += 5; return x; } int bar() { int a = foo() + x + foo(); return a; } What does bar() return? GCC returned 25. Sun’s C compiler returned 20. C says expression evaluation order is implementation-dependent. Side-effects Java prescribes left-to-right evaluation. class Foo { static int x; static int foo() { x += 5; return x; } public static void main(String args[]) { int a = foo() + x + foo(); System.out.println(a); } } Always prints 20. Number Behavior Basic number axioms: a x a if and only if x 0 Additive identity Å Æ Æ (a b) c a (b c) Associative Å Å Æ Å Å a(b c) ab ac Distributive Å Æ Å Misbehaving Floating-Point Numbers 1e20 + 1e-20 = 1e20 1e-20 1e20 ¿ (1 + 9e-7) + 9e-7 1 + (9e-7 + 9e-7) 6Æ 9e-7 1, so it is discarded, however, 1.8e-6 is large enough ¿ 1.00001(1.000001 1) 1.00001 1.000001 1.00001 1 ¡ 6Æ ¢ ¡ ¢ 1.00001 1.000001 1.00001100001 requires too much intermediate ¢ Æ precision.
    [Show full text]