A Right-To-Left Type System for Value Recursion

Total Page:16

File Type:pdf, Size:1020Kb

A Right-To-Left Type System for Value Recursion 1 A right-to-left type system for value recursion 61 2 62 3 63 4 Alban Reynaud Gabriel Scherer Jeremy Yallop 64 5 ENS Lyon, France INRIA, France University of Cambridge, UK 65 6 66 1 Introduction Seeking to address these problems, we designed and imple- 7 67 mented a new check for recursive definition safety based ona 8 In OCaml recursive functions are defined using the let rec opera- 68 novel static analysis, formulated as a simple type system (which we 9 tor, as in the following definition of factorial: 69 have proved sound with respect to an existing operational seman- 10 let rec fac x = if x = 0 then 1 70 tics [Nordlander et al. 2008]), and implemented as part of OCaml’s 11 else x * (fac (x - 1)) 71 type-checking phase. Our check was merged into the OCaml distri- 12 Beside functions, let rec can define recursive values, such as 72 bution in August 2018. 13 an infinite list ones where every element is 1: 73 Moving the check from the middle end to the type checker re- 14 74 let rec ones = 1 :: ones stores the desirable property that compilation of well-typed programs 15 75 Note that this “infinite” list is actually cyclic, and consists of asingle does not go wrong. This property is convenient for tools that reuse 16 76 cons-cell referencing itself. OCaml’s type-checker without performing compilation, such as 17 77 However, not all recursive definitions can be computed. The MetaOCaml [Kiselyov 2014] (which type-checks quoted code) and 18 78 following definition is justly rejected by the compiler: Merlin [Bour et al. 2018] (which type-checks code during editing). 19 79 let rec x = 1 + x Furthermore, some aspects of the check have delicate interactions 20 80 with types, and so cannot be performed on an untyped IR (§4). 21 Here x is used in its own definition. Computing 1 + x requires x to 81 22 have a known value: this definition contains a vicious circle, and 82 any evaluation strategy would fail. Our analysis We looked at reusing existing inference systems, 23 but they do not appear to suit our analysis: they have a finer- 83 24 Functional languages deal with recursive values in various ways. 84 Standard ML simply rejects all recursive definitions except function grained handling of functions and functors than we need, but 25 coarser-grained handling of cyclic data, and most do not propose 85 26 values. At the other extreme, Haskell accepts all well-typed recur- 86 sive definitions, including those that lead to infinite computation. effective inference algorithms. In return for a coarser analysis, our 27 system is noticeably simpler; furthermore, it scales cleanly to the 87 28 In OCaml, safe cyclic-value definitions are accepted, and they are 88 occasionally useful. full OCaml language. 29 A key aspect of our approach is the idea of right-to-left (type to 89 30 For example, consider an interpreter for a programming language 90 with datatypes for ASTs and for values: environment) algorithmic interpretation, which reduces complexity 31 compared to a presentation designed for a left-to-right reading. It is 91 type ast = Fun of var * expr | ::: 32 novel in this space and could inspire other inference rules designers. 92 33 type value = Closure of env * var * expr | ::: 93 34 The eval function builds values from environments and asts 2 Static and dynamic semantics 94 35 let rec eval env = function 95 36 | ::: Syntax Figure1 introduces a minimal subset of ML with the 96 37 | Fun (x, t) -> Closure(env, x, t) interesting ingredients of OCaml’s recursive value definitions: a 97 multi-ary let rec binding let rec ¹x = t ºi in u, functions (λ- 38 Now consider adding an ast constructor FunRec of var * var * expr i i 98 abstractions) λx: t and applications tu, datatype constructors K ¹t ; t ;::: º 39 for recursive functions: FunRec ("f", "x", t) represents the recur- 1 2 99 and shallow pattern-matching match t with ¹K ¹x ºj ! u ºi . 40 sive function let rec f x = t in f . Our OCaml interpreter can i i;j i 100 Other ML constructs (non-recursive let, tuples, conditionals, 41 use value recursion to build a closure for these recursive func- 101 etc.) can be desugared into this core. In fact, the full inference rules 42 tions, without changing the type of the Closure constructor: the 102 for OCaml (and our check) exactly correspond to the rules (and 43 recursive closure simply adds itself to the closure environment 103 check) derived from this desugaring. 44 ((var * value) list). 104 Since ML’s types are largely orthogonal to our analysis, we 45 let rec eval env = function 105 46 present the check using an untyped fragment. (In the full OCaml 106 | ::: language, there are some interactions with types — in particular, 47 | Fun (x, t) -> Closure(env, x, t) 107 48 with GADTs — see §4.) Although we ignore types, we do assume 108 | FunRec (f, x, t) -> i 49 that terms are well-scoped — n.b. in let rec ¹xi = vi º in u, the 109 let rec cl = Closure((f,cl)::env, x, t) in cl i 50 ¹xi º are in scope of u but also of all the vi . 110 51 Our new check and its implementation Until recently, the static 111 Access modes For each recursive binding x = e, our analysis 52 check used by OCaml to reject vicious definitions relied on a syn- 112 assigns an access mode m representing the way that x is accessed 53 tactic analysis, performed on an untyped intermediate language . 113 during evaluation of e. 54 While we believe that the check as originally defined was correct, it 114 Figure2 defines the modes, their order structure, and the mode 55 proved fragile and hard to extend to the interaction of new language 115 composition operations. The modes are as follows: 56 features with recursive definitions. Over the years, bugs were found 116 57 where the check was unduly lenient. In conjunction with OCaml’s Ignore : an expression is entirely unused during the evalua- 117 58 efficient recursive definition compilation scheme [Hirschowitz et al. tion of the program. This is the mode of a variable in an 118 59 2009], this leniency led to segmentation faults. expression in which it does not occur. 119 60 1 120 Alban Reynaud, Gabriel Scherer, and Jeremy Yallop Modes: Ignore ≺ Delay ≺ Guard ≺ Return ≺ Deref 121 Terms 3 t;u ::= x;y; z 122 j let rec b in u Mode composition: 123 j λx: t j t u 0 i m »m ¼ Ignore Delay Guard Return Deref m 124 j K ¹ti º j match t with h Ignore Ignore Ignore Ignore Ignore Ignore 125 Delay Ignore Delay Delay Delay Deref 126 i Guard Ignore Delay Guard Guard Deref 127 Bindings 3 b ::= ¹xi = ti º i Return Ignore Delay Guard Return Deref 128 Handlers 3 h ::= ¹pi ! ti º i Deref Ignore Delay Deref Deref Deref 129 Patterns 3 p; q ::= K ¹xi º 0 m 130 131 Figure 1. Core language syntax Figure 2. Access modes and operations 132 0 133 Γ ` t : m m ≻ m Γ; x : mx ` t : m »Delay¼ Γt ` t : m »Deref¼ Γu ` u : m »Deref¼ 134 Γ ` t : m0 Γ ` λx: t : m Γ + Γ ` t u : m t u 135 136 ¹ ¹ ºj 2I ` ºi 2I ¹ ⪯ ºi;j ¹ ºi ` rec Γi; xj : mi;j ti : Return mi;j Guard xi : Γi b 137 0 Õ h 0i j i 0 i def i i ¹Γi = Γi + ¹mi;j Γj º º ¹mi º = ¹max¹mi; Guardºº Γu; ¹xi : mi º ` u : m 138 0 i 2I i 2I Õ 0 i 139 ¹xi : Γ º ` rec ¹xi = ti º ¹mi »Γi ¼º + Γu ` let rec b in u : m i 140 141 Figure 3. Mode inference rules (abridged) 142 143 Delay : a context can be evaluated (to Weak Normal Form) Finally, Ignore is the absorbing element of mode composition 144 without evaluating its argument. λx: □ is a delay context. (m »Ignore¼ = Ignore = Ignore »m¼), Return is an identity (Return »m¼ = 145 Guard : the context returns the value as a member of a data m = m »Return¼), and composition is idempotent (m »m¼ = m). 146 structure (e.g. a variant or record). K ¹□º is a guard context. 147 The value can safely be defined mutually-recursively with A right-to-left inference system Figure3 gives a representative 148 its context, as in let rec x = K ¹xº1. sample of the inference rules for a judgment of form Γ ` t : m 149 Return : the context returns its value without further inspec- for term t, access mode m and environment Γ that maps term vari- 150 tion. This value cannot be defined mutually-recursively with ables to access modes. Modes classify terms and variables, playing 151 its context, to avoid self-loops: in let rec x = x and let rec x = the role of types in usual type systems. The example judgment 152 let y = x in y, the last occurrence of x is in Return context. x : Deref;y : Delay ` ¹x + 1; lazy yº : Guard can be read either 153 154 Deref : the context inspects and uses the value in arbitrary left-to-right: If x can safely be used in Deref mode, and y in 155 ways. Such a value must be fully defined at the point of Delay mode, then ¹x + 1; lazy yº can safely be used at Guard. 156 usage; it cannot be defined mutually-recursively with its right-to-left: If a context accesses the term ¹x+1; lazyyº under 157 context.
Recommended publications
  • Unfold/Fold Transformations and Loop Optimization of Logic Programs
    Unfold/Fold Transformations and Loop Optimization of Logic Programs Saumya K. Debray Department of Computer Science The University of Arizona Tucson, AZ 85721 Abstract: Programs typically spend much of their execution time in loops. This makes the generation of ef®cient code for loops essential for good performance. Loop optimization of logic programming languages is complicated by the fact that such languages lack the iterative constructs of traditional languages, and instead use recursion to express loops. In this paper, we examine the application of unfold/fold transformations to three kinds of loop optimization for logic programming languages: recur- sion removal, loop fusion and code motion out of loops. We describe simple unfold/fold transformation sequences for these optimizations that can be automated relatively easily. In the process, we show that the properties of uni®cation and logical variables can sometimes be used to generalize, from traditional languages, the conditions under which these optimizations may be carried out. Our experience suggests that such source-level transformations may be used as an effective tool for the optimization of logic pro- grams. 1. Introduction The focus of this paper is on the static optimization of logic programs. Speci®cally, we investigate loop optimization of logic programs. Since programs typically spend most of their time in loops, the generation of ef®cient code for loops is essential for good performance. In the context of logic programming languages, the situation is complicated by the fact that iterative constructs, such as for or while, are unavailable. Loops are usually expressed using recursive procedures, and loop optimizations have be considered within the general framework of inter- procedural optimization.
    [Show full text]
  • Functional Programming (ML)
    Functional Programming CSE 215, Foundations of Computer Science Stony Brook University http://www.cs.stonybrook.edu/~cse215 Functional Programming Function evaluation is the basic concept for a programming paradigm that has been implemented in functional programming languages. The language ML (“Meta Language”) was originally introduced in the 1970’s as part of a theorem proving system, and was intended for describing and implementing proof strategies. Standard ML of New Jersey (SML) is an implementation of ML. The basic mode of computation in SML is the use of the definition and application of functions. 2 (c) Paul Fodor (CS Stony Brook) Install Standard ML Download from: http://www.smlnj.org Start Standard ML: Type ''sml'' from the shell (run command line in Windows) Exit Standard ML: Ctrl-Z under Windows Ctrl-D under Unix/Mac 3 (c) Paul Fodor (CS Stony Brook) Standard ML The basic cycle of SML activity has three parts: read input from the user, evaluate it, print the computed value (or an error message). 4 (c) Paul Fodor (CS Stony Brook) First SML example SML prompt: - Simple example: - 3; val it = 3 : int The first line contains the SML prompt, followed by an expression typed in by the user and ended by a semicolon. The second line is SML’s response, indicating the value of the input expression and its type. 5 (c) Paul Fodor (CS Stony Brook) Interacting with SML SML has a number of built-in operators and data types. it provides the standard arithmetic operators - 3+2; val it = 5 : int The Boolean values true and false are available, as are logical operators such as not (negation), andalso (conjunction), and orelse (disjunction).
    [Show full text]
  • Abstract Recursion and Intrinsic Complexity
    ABSTRACT RECURSION AND INTRINSIC COMPLEXITY Yiannis N. Moschovakis Department of Mathematics University of California, Los Angeles [email protected] October 2018 iv Abstract recursion and intrinsic complexity was first published by Cambridge University Press as Volume 48 in the Lecture Notes in Logic, c Association for Symbolic Logic, 2019. The Cambridge University Press catalog entry for the work can be found at https://www.cambridge.org/us/academic/subjects/mathematics /logic-categories-and-sets /abstract-recursion-and-intrinsic-complexity. The published version can be purchased through Cambridge University Press and other standard distribution channels. This copy is made available for personal use only and must not be sold or redistributed. This final prepublication draft of ARIC was compiled on November 30, 2018, 22:50 CONTENTS Introduction ................................................... .... 1 Chapter 1. Preliminaries .......................................... 7 1A.Standardnotations................................ ............. 7 Partial functions, 9. Monotone and continuous functionals, 10. Trees, 12. Problems, 14. 1B. Continuous, call-by-value recursion . ..................... 15 The where -notation for mutual recursion, 17. Recursion rules, 17. Problems, 19. 1C.Somebasicalgorithms............................. .................... 21 The merge-sort algorithm, 21. The Euclidean algorithm, 23. The binary (Stein) algorithm, 24. Horner’s rule, 25. Problems, 25. 1D.Partialstructures............................... ......................
    [Show full text]
  • Generating Mutually Recursive Definitions
    Generating Mutually Recursive Definitions Jeremy Yallop Oleg Kiselyov University of Cambridge, UK Tohoku University, Japan [email protected] [email protected] Abstract let rec s = function A :: r ! s r j B :: r ! t r j [] ! true and t = function A :: r ! s r j B :: r ! u r j [] ! false Many functional programs — state machines (Krishnamurthi and u = function A :: r ! t r j B :: r ! u r j [] ! false 2006), top-down and bottom-up parsers (Hutton and Meijer 1996; Hinze and Paterson 2003), evaluators (Abelson et al. 1984), GUI where each function s, t, and u realizes a recognizer, taking a list of initialization graphs (Syme 2006), &c. — are conveniently ex- A and B symbols and returning a boolean. However, the program pressed as groups of mutually recursive bindings. One therefore that builds such a group from a description of an arbitrary state expects program generators, such as those written in MetaOCaml, machine cannot be expressed in MetaOCaml. to be able to build programs with mutual recursion. The limited support for generating mutual recursion is a con- Unfortunately, currently MetaOCaml can only build recursive sequence of expression-based quotation: brackets enclose expres- groups whose size is hard-coded in the generating program. The sions, and splices insert expressions into expressions — but a group general case requires something other than quotation, and seem- of bindings is not an expression. There is a second difficulty: gen- ingly weakens static guarantees on the resulting code. We describe erating recursive definitions with ‘backward’ and ‘forward’ refer- the challenges and propose a new language construct for assuredly ences seemingly requires unrestricted, Lisp-like gensym, which de- generating binding groups of arbitrary size – illustrating with a col- feats MetaOCaml’s static guarantees.
    [Show full text]
  • Inductively Defined Functions in Functional Programming Languages
    CORE Metadata, citation and similar papers at core.ac.uk Provided by Elsevier - Publisher Connector JOURNAL OF COMPUTER AND SYSTEM SCIENCES 34, 4099421 (1987) Inductively Defined Functions in Functional Programming Languages R. M. BURSTALL Department of Computer Science, University of Edinburgh, King$ Buildings, Mayfield Road, Edinburgh EH9 3JZ, Scotland, U.K. Received December 1985; revised August 1986 This paper proposes a notation for defining functions or procedures in such a way that their termination is guaranteed by the scope rules. It uses an extension of case expressions. Suggested uses include programming languages and logical languages; an application is also given to the problem of proving inequations from initial algebra specifications. 0 1987 Academic Press, Inc. 1. INTRODUCTION When we define recursive functions on natural numbers we can ensure that they terminate without a special proof, by adhering to the schema for primitive recur- sion. The schema can be extended to other data types. This requires inspection of the function definition to ensure that it conforms to such a schema, involving second-order pattern matching. However, if we combine the recursion with a case expression which decomposes elements of the data type we can use the ordinary scoping rules for variables to ensure termination, without imposing any special schema. I formulate this proposal for “inductive” case expressions in ML [S], since it offers succinct data-type definitions and a convenient case expression. Any other functional language with these facilities could do as well. A number of people have advocated the use of initial algebras to define data types in specification languages, see, for example, Goguen, Thatcher and Wagner [3] and Burstall and Goguen [ 11.
    [Show full text]
  • Using Prolog for Transforming XML Documents
    Using Prolog for Transforming XML Documents Rene´ Haberland Technical University of Dresden, Free State of Saxony, Germany Saint Petersburg State University, Saint Petersburg, Russia email: [email protected] (translation into English from 2007) Abstract—Proponents of the programming language Prolog all or with severe restrictions into hosting languages, like Java, share the opinion Prolog is more appropriate for transforming C++ or Pascal. XML documents than other well-established techniques and In order to resolve this problem, two strategies can be languages like XSLT. This work proposes a tuProlog-styled interpreter for parsing XML documents into Prolog-internal identified as most promising. First, integrate new features. The lists and vice versa for serialising lists into XML documents. language gets extended. However, this can only succeed if lingual concepts are universal enough w.r.t. lexemes, idioms. Based on this implementation, a comparison between XSLT Second, choose a federated approach. Depending on the imple- and Prolog follows. First, criteria are researched, such as con- mentation, the hosting language is simulated by introspection. sidered language features of XSLT, usability and expressibility. These criteria are validated. Second, it is assessed when Prolog Solely concepts remain untouched. distinguishes between input and output parameters towards The reasons against the first approach are a massive rise in reversible transformation. complexity and a notoriously incompatible paradigm. Hence, the federated approach is chosen to implement the transfor- I. INTRODUCTION mation language with Prolog being visible to the user. A. Preliminaries A transformation within XML is a mapping from XML onto B. Motivation XML. W.l.o.g. only XML as output is considered in this work.
    [Show full text]
  • Recursion: Memoization Introduction to Object Oriented Programming Instructors: Daniel Deutch , Amiram Yehudai Teaching Assistants: Michal Kleinbort, Amir Rubinstein
    Extended Introduction to Computer Science CS1001.py Lecture 13: More Recursion: Memoization Introduction to Object Oriented Programming Instructors: Daniel Deutch , Amiram Yehudai Teaching Assistants: Michal Kleinbort, Amir Rubinstein School of Computer Science Tel-Aviv University Winter Semester, 2013-15 http://tau-cs1001-py.wikidot.com Lecture 11-12 : Highlights Recursion, and recursive functions • Basic examples and definition of recursion • Fibonacci • factorial • Binary search - revisited • Sorting • QuickSort • MergeSort • Towers of Hanoi • Towers of Hanoi nightmare • Tail recursion 2 Lecture 13 - Plan Recursion, and recursive functions Revisiting the results of fibonacci Memoization Recursion depth limit Two additional examples for recursion: Ackermann function Mutual recursion Classes and methods (a very gentle intro to object oriented programming). 3 Pitfalls of Using Recursion Every modern programming language, including, of course, Python, supports recursion as one of the built- in control mechanism. However, recursion is not the only control mechanism in Python, and surely is not the one employed most often. Furthermore, as we will now see, cases where “naive recursion" is highly convenient for writing code may lead to highly inefficient run times. For this reason, we will also introduce techniques to get rid of recursion (in addition to the elimination of tail recursion that we already saw). We note, however, that in some cases, eliminating recursion altogether requires very crude means. 4 Computing Fibonacci Numbers We coded Fibonacci numbers, using recursion, as following: def fibonacci (n): """ plain Fibonacci , using recursion """ if n <2: return 1 else : return fibonacci (n -2)+ fibonacci (n -1) But surely nothing could go wrong with such simple and elegant code..
    [Show full text]
  • Strong Functional Pearl: Harper's Regular-Expression Matcher In
    Strong Functional Pearl: Harper’s Regular-Expression Matcher in Cedille AARON STUMP, University of Iowa, United States 122 CHRISTOPHER JENKINS, University of Iowa, United States of America STEPHAN SPAHN, University of Iowa, United States of America COLIN MCDONALD, University of Notre Dame, United States This paper describes an implementation of Harper’s continuation-based regular-expression matcher as a strong functional program in Cedille; i.e., Cedille statically confirms termination of the program on all inputs. The approach uses neither dependent types nor termination proofs. Instead, a particular interface dubbed a recursion universe is provided by Cedille, and the language ensures that all programs written against this interface terminate. Standard polymorphic typing is all that is needed to check the code against the interface. This answers a challenge posed by Bove, Krauss, and Sozeau. CCS Concepts: • Software and its engineering ! Functional languages; Recursion; • Theory of compu- tation ! Regular languages. Additional Key Words and Phrases: strong functional programming, regular-expression matcher, programming with continuations, recursion schemes ACM Reference Format: Aaron Stump, Christopher Jenkins, Stephan Spahn, and Colin McDonald. 2020. Strong Functional Pearl: Harper’s Regular-Expression Matcher in Cedille. Proc. ACM Program. Lang. 4, ICFP, Article 122 (August 2020), 25 pages. https://doi.org/10.1145/3409004 1 INTRODUCTION Constructive type theory requires termination of programs intended as proofs under the Curry- Howard isomorphism [Sørensen and Urzyczyn 2006]. The language of types is enriched beyond that of traditional pure functional programming (FP) to include dependent types. These allow the expression of program properties as types, and proofs of such properties as dependently typed programs.
    [Show full text]
  • A Survey on Teaching and Learning Recursive Programming
    Informatics in Education, 2014, Vol. 13, No. 1, 87–119 87 © 2014 Vilnius University A Survey on Teaching and Learning Recursive Programming Christian RINDERKNECHT Department of Programming Languages and Compilers, Eötvös Loránd University Budapest, Hungary E-mail: [email protected] Received: July 2013 Abstract. We survey the literature about the teaching and learning of recursive programming. After a short history of the advent of recursion in programming languages and its adoption by programmers, we present curricular approaches to recursion, including a review of textbooks and some programming methodology, as well as the functional and imperative paradigms and the distinction between control flow vs. data flow. We follow the researchers in stating the problem with base cases, noting the similarity with induction in mathematics, making concrete analogies for recursion, using games, visualizations, animations, multimedia environments, intelligent tutor- ing systems and visual programming. We cover the usage in schools of the Logo programming language and the associated theoretical didactics, including a brief overview of the constructivist and constructionist theories of learning; we also sketch the learners’ mental models which have been identified so far, and non-classical remedial strategies, such as kinesthesis and syntonicity. We append an extensive and carefully collated bibliography, which we hope will facilitate new research. Key words: computer science education, didactics of programming, recursion, tail recursion, em- bedded recursion, iteration, loop, mental models. Foreword In this article, we survey how practitioners and educators have been teaching recursion, both as a concept and a programming technique, and how pupils have been learning it. After a brief historical account, we opt for a thematic presentation with cross-references, and we append an extensive bibliography which was very carefully collated.
    [Show full text]
  • One with RECURSIVE Is Worth Many Gotos
    One WITH RECURSIVE is Worth Many GOTOs Denis Hirn Torsten Grust University of Tübingen Tübingen, Germany [denis.hirn,torsten.grust]@uni-tuebingen.de ABSTRACT Figure 1c shows a selection of source/dest pairs and the paths PL/SQL integrates an imperative statement-by-statement style of computed by route, assuming generous ttl budgets. programming with the plan-based evaluation of SQL queries. The Observe how PL/SQL UDF route embeds several scalar SQL ex- disparity of both leads to friction at runtime, slowing PL/SQL ex- pressions like loc <> dest, ttl < hop.cost, or route || loc (see ecution down significantly. This work describes a compiler from Lines 10, 14, and 18 in Figure 2). PostgreSQL’s PL/SQL interpreter PL/SQL UDFs to plain SQL queries. Post-compilation, evaluation en- evaluates these simple SQL expressions via a fast path that directly tirely happens on the SQL side of the fence. With the friction gone, invokes the system’s SQL expression evaluator. we observe execution times to improve by about a factor of 2, even Notably, though, route also contains the SQL SELECT-block Q1 for complex UDFs. The compiler builds on techniques long estab- which the function uses to query the routing table for the next hop lished by the programming language community. In particular, it from loc towards dest (the two PL/SQL variables loc and dest are uses trampolined style to compile arbitrarily nested iterative con- free in the SELECT-block and act like parameters for Q1, see Line 13 trol flow in PL/SQL into SQL’s recursive common table expressions.
    [Show full text]
  • Recursive Function Definitions for Isabelle/HOL
    FAKULTÄT FÜR INFORMATIK DER TECHNISCHEN UNIVERSITÄT MÜNCHEN Bachelor’s Thesis in Computer Science Primitively (co)recursive function definitions for Isabelle/HOL Lorenz Panny FAKULTÄT FÜR INFORMATIK DER TECHNISCHEN UNIVERSITÄT MÜNCHEN Bachelor’s Thesis in Computer Science Primitively (co)recursive function definitions for Isabelle/HOL Primitiv-(ko)rekursive Funktionsdefinitionen für Isabelle/HOL Author: Lorenz Panny Supervisor: Prof. Tobias Nipkow, Ph.D. Advisor: Dr. Jasmin Blanchette Advisor: Dmitriy Traytel Date: August 15, 2014 I assure the single-handed composition of this thesis only supported by declared resources. Abstract Isabelle/HOL has lately been extended with a definitional package supporting modular (co)datatypes based on category theoretical constructions. The implementation generates the specified types and associated theorems and constants, notably (co)recursors, but ini- tially, there was no convenient way of specifying functions over these types. This thesis introduces the high-level commands primrec, primcorec and primcorecursive that can be used to define primitively (co)recursive functions over Isabelle’s new (co)datatypes using an intuitive syntax. Automating a tedious process, a user specification is internally re- duced to a (co)recursor-based definition. Using the (co)recursor theorems, it is then proved and introduced as theorems that the definition does in fact fulfill the specified properties. Zusammenfassung Isabelle/HOL wurde kürzlich um ein definitorisches Modul zur Unterstützung modula- rer Ko-/Datentypen, basierend auf einer kategorientheoretischen Konstruktion, erweitert. Die Implementierung generiert die spezifizierten Typen sowie zugehörige Theoreme und Konstanten (insbesondere Ko-/Rekursoren), aber zunächst stand keine bequeme Methode zum Erzeugen von Funktionen über diesen Typen zur Verfügung. In dieser Arbeit werden die Befehle primrec, primcorec und primcorecursive vorgestellt, mit deren Hilfe unter Benutzung intuitiver Syntax primitiv-(ko)rekursive Funktionen über Isabelles neuen Ko- /Datentypen definiert werden können.
    [Show full text]
  • Compiling PL/SQL Away
    Compiling PL/SQL Away Christian Duta Denis Hirn Torsten Grust Universität Tübingen Tübingen, Germany [ christian.duta, denis.hirn, torsten.grust ]@uni-tuebingen.de ABSTRACT Context switches will be abundant. If f’s call site is lo- SELECT FROM WHERE Q “PL/SQL functions are slow,” is common developer wisdom cated inside a - - block of , each row pro- f f that derives from the tension between set-oriented SQL eval- cessed by the block will invoke . Likewise, if embeds e.g. FOR uation and statement-by-statement PL/SQL interpretation. multiple queries or employs iteration, , in terms of WHILE Q We pursue the radical approach of compiling PL/SQL away, or loops, we observe repeated plan evaluation for the i. turning interpreted functions into regular subqueries that Unfortunately, both kinds of context switches are costly. can then be efficiently evaluated together with their em- Each switch Q→f incurs overhead for PL/SQL interpreter invo- bracing SQL query, avoiding any PL/SQL ↔ SQL context cation or resumption. A switch f→Qi leads to overhead due switches. Input PL/SQL functions may exhibit arbitrary to (1) plan generation and caching on the first evaluation control flow. Iteration, in particular, is compiled into SQL- of Qi or (2) plan cache lookup, plan instantiation, and plan level recursion. RDBMSs across the board reward this com- teardown for each subsequent evaluation of Qi. Iteration in pilation effort with significant run time savings that render both, Q and f, multiplies the toll. established developer lore questionable. Let us make the conundrum concrete with PL/pgSQL func- tion walk() of Figure 3.
    [Show full text]