Programming Computable Functions

En informatique théorique, Programming Computable Functions ou PCF est un langage de programmation théorique apparu pour la première fois en 1977 dans un article de Gordon Plotkin[1], mais qui est basé sur des notes de Dana S. Scott de 1969 qui n'ont été publiées qu'en 1993[2]. Ce langage consiste en une extension du lambda-calcul simplement typé avec des combinateurs de points fixes et des entiers naturels, ce qui permet de récupérer la complétude au sens de Turing, c'est-à-dire la possibilité d'exprimer n'importe quelle fonction calculable. En effet, l'ajout d'une discipline de types au lambda-calcul, donnant le lambda-calcul simplement typé, limite l'expressivité du langage et les fonctions qui y sont définissables ; l'ajout de combinateurs de point fixe et d'entiers résout ce problème.

Syntaxe et typage

PCF reprend les types du lambda-calcul simplement typé en y adjoignant un type représentant comme types de base les booléens, , et un type représentant les entiers, [1],[3]. Les types sont donc , , ou avec et des types déjà formés. Ce type représente le type des fonctions de vers . La notation doit être lue comme .

De plus, la syntaxe de PCF comprend la syntaxe du lambda-calcul simplement typé, et les constructions de base y sont les mêmes : il y a des variables, notées  ; si est un terme de type et un terme de type , est un terme de type qui représente l'évaluation de la fonction en  ; et si est un terme de type dans un contexte où la variable est de type , est de type et représente la fonction . Si est un terme de type , on notera . Pour plus de détails, consulter les sections Syntaxe et Règles de typage de l'article sur le lambda-calcul simplement typé.

À cela se rajoutent les constantes propres permettant de manipuler les types de base. Pour chaque entier naturel , on ajoute une constante . De plus, il y a une constante pour chaque booléen, et . Pour chaque type de base , c'est-à-dire et , on a une constante représentant une instruction conditionnelle : l'idée étant que si vaut , l'expression vaut , si vaut , elle s'évalue en . De plus, il y a deux fonctions pour manipuler les entiers, la fonction successeur et la fonction prédécesseur , qui correspondent à ajouter ou soustraire à leur argument. Enfin, il y a une fonction qui renvoie si son argument vaut l'entier , sinon.

Enfin, le principal ingrédient de PCF est sa construction de point fixe : pour chaque type , on dispose d'une constante dont l'interprétation est qu'il renvoie le plus petit point fixe de la fonction qu'on lui passe en argument. Certains auteurs[4] rajoutent à chaque type une constante qui répresente un programme qui ne termine pas. Cette construction n'est pas incluse dans la présentation originelle de Gordon Plotkin[1], mais peut y être définie comme .

Sémantique opérationnelle

PCF peut être muni d'une sémantique opérationnelle [5]. Si et sont des termes, on a si se transforme en après une étape élémentaire de calcul. On note pour dire que se transforme en en zéro ou plusieurs étapes. La relation est définie comme suit :

  •  : pour évaluer l'application de la fonction en , on remplace par dans  ;
  •  : cela exprime que est un point fixe de  ;
  • ;
  •  ;
  • ;
  •  ;
  •  ;
  •  ;
  • Les règles contextuelles sont les suivantes : si alors , , , et . Elles donnnent à PCF une sémantique d'appel par nom.

Cette sémantique est déterministe, c'est-à-dire que si et , alors . De plus, elle préserve le typage : si et que est de type , alors aussi.

De plus, on pourrait autoriser la réduction dans tous les contextes, plutôt qu'uniquement dans ceux précisés dans la dernière règle (qui interdit, par exemple, la réduction avec et qui n'est pas une lambda-abstraction ou ou ou ). Dans ce cas, la réduction obtenue préserve encore le typage et est confluente. De plus, si avec cette réduction, et si est en forme normale, alors avec la version restreinte. En clair, la sémantique opérationnelle définie ici est standardisante[6].

Exemple

PCF permet par exemple de définir des programmes effectuant l'addition de deux entiers. En voici un[6] :

.

Si on note la fonction à l'intérieur de , la somme de deux et trois est calculée par la suite de réductions suivante :

.

Ainsi, se réduit bien en la valeur .

Sémantique dénotationnelle

La sémantique opérationnelle présentée plus haut permet de déterminer comment exécuter un terme de PCF comme un programme. Mais pour étudier PCF, il peut être interéssant de traduire un terme comme en une fonction au sens mathématique du terme. C'est le rôle de la sémantique dénotationnelle[7]. Mais contrairement au lambda-calcul simplement typé, on ne peut pas interpréter PCF directement dans les fonctions et les ensembles, puisqu'il existe des fonctions qui n'ont pas de point fixe. On va donc interpréter les termes de PCF comme des fonctions continues entre cpo. Un cpo est un ensemble ordonné dans lequel toutes les parties dirigées ont une borne supérieure, et qui possède un plus petit élément[8]. L'intérêt des cpo réside dans ce cas chaque fonction continue possède un plus petit point fixe. La catégorie des cpo est cartésienne fermée[9].

Modèle continu

Le modèle continu est l'interprétation usuelle de PCF[10],[11]. On va définir pour chaque type, chaque contexte et chaque terme une traduction telle que l'interprétation d'un contexte ou d'un type soit un cpo, et si dans PCF, alors est une fonction continue de l'interprétation du contexte vers l'interprétation du type de .

Les types de base sont interprétés comme des domaines plats : on a et , où est l'ensemble des entiers naturels, l'ensemble des booléens et pour tout ensemble , est le cpo défini sur l'ensemble auquel on adjoint un élément , avec l'ordre défini par pour tout , et les éléments de sont incomparables entre eux. Le type est interprété comme l'ensemble des fonctions continues de vers . Un contexte est interprété par . On remarque que chaque type est un cpo, donc a un plus petit élément .

Considérons maintenant un terme typé et son contexte , avec .

L'interprétation des constructions issues du lambda-calcul est la suivante :

  • .
  • .
  • .

L'opérateur de point fixe est définie comme suit :

  • est la fonction qui à chaque associe son plus petit point fixe. Il est défini comme , c'est-à-dire comme le supremum de la suite croissante .
  • Puisque intuitivement, , et que le plus petit point fixe de la fonction identité est , .

Les primitives sur les entiers et les booléens sont interprétées comme suit, l'idée étant que représente une valeur non déterminée, par exemple un programme qui ne termine pas :

  • si et .
  • si et , et .
  • , si et , et .
  • , et .

Modèles standards

Plus généralement, on peut interpréter PCF dans des catégories cartésiennes fermées enrichies dans les cpo. Un tel modèle est appelé modèle standard de PCF. Une catégorie cartésienne fermée enrichie dans les cpo est une catégorie cartésienne fermée telle que :

  • les hom-set sont des cpo pour tous objets et  ;
  • la composition, le pairage et la currification sont continues ;
  • l'évaluation et la composition sont strictes : pour tous objets , et et tout morphisme , si l'on désigne par le plus petit élément d'un cpo , on a :
    ,
    et si l'on désigne par l'objet des fonctions de vers dans la catégorie cartésienne fermée, par l'opération de pairage et par le morphisme d'évaluation, on a :
.
  • la catégorie possède deux objets et tels que soit isomorphe à et soit isomorphe à en tant que cpo, où l'objet terminal de la catégorie . Dans la catégorie des cpo, est l'ensemble à un élément muni de l'égalité comme relation d'ordre.

Dans ce cas, en interprétant dans les constructions issues du lambda-calcul simplement typé de la façon uselle, en définissant et et en interprétant les constructions sur les types primitifs de la même façon que dans le modèle continu en posant et , ce qui est rendu possible par les isomorphismes et , on obtient la sémantique désirée.

Expressivité

Turing-complétude

Si est un programme dans PCF qui prend entiers en entrée et renvoie un entier, on peut montrer assez facilement que détermine une fonction calculable par si et n'est pas définie sinon.

Réciproquement, étant donnée une fonction calculable , on peut se demander si elle est représentée par un terme de PCF tel que pour tous entiers et ,

.

Ce problème correspond à la notion de complétude au sens de Turing. La réponse est positive, donc PCF peut représenter toutes les fonctions calculables entre les entiers[12].

Bibliographie

Notes et références

  1. a b et c (en) Gordon Plotkin, « LCF considered as a programming language », Theoretical Computer Science, vol. 5, no 3,‎ , p. 223–255 (DOI 10.1016/0304-3975(77)90044-5 Accès libre, lire en ligne Accès libre [PDF])
  2. (en) Dana S. Scott, « A type-theoretical alternative to ISWIM, CUCH, OWHY », Theoretical Computer Science, vol. 121, no 1,‎ , p. 411–440 (ISSN 0304-3975, DOI 10.1016/0304-3975(93)90095-B Accès libre, lire en ligne Accès libre [PDF], consulté le ) — distribué à l'origine comme des notes non publiées d'un séminaire donné à Oxford en 1969 sous le nom A theory of computable functions of higher type.
  3. Amadio et Curien 1998, p. 149
  4. Amadio et Curien 1998, p. 145
  5. Amadio et Curien 1998, p. 151
  6. a et b Amadio et Curien 1998, p. 150
  7. Amadio et Curien 1998, p. 5
  8. Amadio et Curien 1998, p. 14-15
  9. Amadio et Curien 1998, p. 95
  10. Amadio et Curien 1998, p. 146
  11. Amadio et Curien 1998, p. 149-150
  12. Scott 1993, p. 435