Section courante

A propos

Section administrative du site

 Langage  Programmation  Annexe  Aide 
ABAP/4
Ada
Assembleur
Assembly & bytecode
ASP (Active Server Pages)
Basic
C
C++
C# (C Sharp)
Cobol
ColdFusion
Fortran
HTML
Java
JavaScript
LISP
Logo
LotusScript
Oberon
Pascal
Perl
PHP
PL/1
Prolog
Python
Rebol
REXX
Ruby
SAS
NoSQL
SQL
Swift
X++ (Axapta)
GNAT
SMALLAda
VHDL
Assembleur 370
Assembleur 1802
Assembleur 4004
Assembleur 6502
Assembleur 6800
Assembleur 68000
Assembleur 8080 et 8085
Assembleur 8089
Assembleur 80x86
Assembleur AGC4
Assembleur ARM
Assembleur DPS 8000
Assembleur i860
Assembleur Itanium
Assembleur MIPS
Assembleur PDP-11
Assembleur PowerPC
Assembleur RISC-V
Assembleur SPARC
Assembleur SuperH
Assembleur UNIVAC I
Assembleur VAX
Assembleur Z80
Assembleur Z8000
Assembleur z/Architecture
ASSEMBLER/MONITOR 64
Micol Assembler
GFA Assembler
A86
MASM (Macro Assembler)
TASM (Turbo Assembler)
CIL
Jasmin
LLVM
MSIL
Parrot
P-Code (PCode)
SWEET16
G-Pascal
ASP 1.0
ASP 2.0
ASP 3.0
ASP.NET
ASP.NET Core
ABasiC (Amiga)
Adam SmartBASIC
Altair BASIC
AmigaBASIC (Amiga)
AMOS Basic (Amiga)
Atari Basic (Atari 400, 600 XL, 800, 800XL)
Basic Apple II (Integer BASIC/APPLESOFT)
Basic Commodore 64 (CBM-BASIC)
Basic Commodore 128 (BASIC 7.0)
Basic Commodore VIC-20 (CBM-BASIC 2.0)
Basic Coco 1 (Color Basic)
Basic Coco 2 (Extended Color Basic)
Basic Coco 3 (Extended Color Basic 2.0)
BASICA (PC DOS)
Basic Pro
BBC BASIC
Blitz BASIC (Amiga)
DarkBASIC
Dartmouth BASIC
GFA-Basic (Atari ST/Amiga)
GWBASIC (MS-DOS)
Liberty BASIC
Locomotive BASIC (Amstrad CPC)
MSX-Basic
Omikron Basic (Atari ST)
Oric Extended Basic
Power Basic
Quick Basic/QBasic (MS-DOS)
Sinclair BASIC (ZX80, ZX81, ZX Spectrum)
ST BASIC (Atari ST)
Turbo Basic
Vintage BASIC
VBScript
Visual Basic (VB)
Visual Basic .NET (VB .NET)
Visual Basic pour DOS
Yabasic
BeckerBASIC
SIMONS' BASIC
Basic09 d'OS-9
Disk Extended Color Basic
Basic09 d'OS-9
Disk Extended Color Basic
Access
Excel
Visual Basic pour Windows
Visual Basic .NET pour Windows
C Shell Unix (csh)
C pour Amiga
C pour Atari ST
C pour DOS
C pour Falcon030
C pour GEMDOS (Atari ST)
C pour Linux
C pour PowerTV OS
C pour OS/2
C pour Unix
C pour Windows
Aztec C
CoCo-C
GNU C
HiSoft C
IBM C/2
Introl-C
Lattice C
Microsoft C
MinGW C
MSX-C
Open Watcom C
OS-9 C Compiler
Pure C
Quick C
Turbo C
HiSoft C for Atari ST
HiSoft C for CP/M (Amstrad CPC)
C++ pour OS/2
C++ pour Windows
Borland C++
C++Builder
IBM VisualAge C++
Intel C++
MinGW C++
Open Watcom C++
Symantec C++
Turbo C++
Visual C++
Visual C++ .NET
Watcom C++
Zortech C++
C# (C Sharp) pour Windows
Apple III Cobol
Microsoft Cobol
BlueDragon
Lucee
OpenBD
Railo
Smith Project
Microsoft Fortran
WATFOR-77
CSS
FBML
Open Graph
SVG
XML
XSL/XSLT
LESS
SASS
GCJ (GNU)
JSP
Jython
Visual J++
Node.js
TypeScript
AutoLISP
ACSLogo
LotusScript pour Windows
Amiga Oberon
Oberon .NET
Apple Pascal
Delphi/Kylix/Lazarus
Free Pascal
GNU Pascal
HighSpeed Pascal
IBM Personal Computer Pascal
Lisa Pascal
Maxon Pascal
MPW Pascal
OS-9 Pascal
OSS Personal Pascal
Pascal-86
Pascal du Cray Research
Pascal/VS
Pascal-XT
PURE Pascal
QuickPascal
RemObjets Chrome
Sun Pascal
THINK Pascal
Tiny Pascal (TRS-80)
Turbo Pascal
UCSD Pascal
VAX Pascal
Virtual Pascal
Turbo Pascal for CP/M-80
Turbo Pascal for DOS
Turbo Pascal for Macintosh
Turbo Pascal for Windows
CodeIgniter (Cadre d'application)
Drupal (Projet)
Joomla! (Projet)
Phalanger (PHP .NET)
phpBB (Projet)
Smarty (balise)
Twig (balise)
Symfony (Cadre d'application)
WordPress (Projet)
Zend (Cadre d'application)
PL360
PL/M-80
PL/M-86
Turbo Prolog
CPython
IronPython
Jython
PyPy
AREXX
Regina REXX
JMP
Btrieve
Cassandra
Clipper
CouchDB
dBASE
Hbase
Hypertable
MongoDB
Redis
Access
BigQuery
DB2
H2
Interbase
MySQL
Oracle
PostgreSQL
SAP HANA
SQL Server
Sybase
U-SQL
Affichage
Clavier
Cyrix
Traitement de texte
Manette de jeux
Mathématique
Modem
Partition
Presse-Papier
Son
Souris
Système d'exploitation
Hacker
Introduction
Métaphysique de l'affichage
Affichage d'élément simple
Traitement des couleurs
Accélérateur graphiques
Technique d'affichage de forme
Les Scalaires (images sur mesure)
Les Fractales
BIOS CGA pour Graphic Solutions
Triangle de «Sierpinski»
PS/2
Exécution de programmes fils et de sessions
Bibliothèque de code
Sources
Lexique et dictionnaire 80x86
Code Assembleur
Extension Fichier
Interruptions
Port d'E/S
RAM
Table Code
8080 et 8085
68000
Java
00h à 07h: Processeur
08h à 0Fh: Périphériques (IRQ0 à IRQ7)
10h: Gestionnaire vidéo
13h: Accès aux disques
15h: Extension d'Entrée/Sortie
16h: Clavier
21h: Service DOS
2Ah: Interface de service réseau
2Fh: Multiplexe
33h: Interface du pilote de souris
59h: GSS Computer Graphics Interface
5Ch: Réseau BIOS (NetBIOS)
67h: Gestionnaire EMM/EMS
00h: Mode vidéo
01h: Sélection du type de curseur
10h, F0h: Tseng Labs HiColor
4Fh: Pilote VESA
5Fh: Realtek RTVGA
6Fh: Video 7
18h: Demande des informations SCSI
01h: Écrit dans la RAM non-volatile (Amstrad PC1512)
02h: Lecture dans la RAM non-volatile (Amstrad PC1512)
10h: Gestionnaire principal TopView
89h: Entrée en mode protégé
00h: Lecture d'un caractère
64h: OS/2 API DOS32StartSession
FFh, 92h: Détecte la présence du pilote «PREDIR.EXE»
16h: Windows & DPMI
0Ah: Détection de Windows
84h: Demande le point d'entrée
86h: Détection du mode DPMI
87h: Détecte l'installation DPMI
8Eh: Machine virtuel et application
APPLE ][
ATARI 2600
Commodore 64
IBM PC
Préface
Notes légal
Dictionnaire
Recherche

Sommaire


Introduction

Cette page présente différentes résolutions de problèmes mathématiques à l'aide du Turbo Pascal et de l'Assembleur :

Multiplication 32 bits en 16 bits

Malgré les apparences, il n'est pas impossible d'effectuer des multiplications en 32 bits sur un processeurs 16 bits comme un 8088 et 8086. Pour ce faire, il suffit d'effectuer les opérations en deux parties (par tranche de 16 bits) de les additions à la partie base ou haute (dépendamment de la tranche en question).

Ces opérations est donc équivalente à DX:AX = a x b:

LES AX,a                           ; Valeur a
MOV DX,ES
LES CX,b                           ; Valeur b
MOV BX,ES
MOV SI,AX
MOV DI,DX
MUL CX
PUSH AX
PUSH DX
  MOV AX,SI
  MUL BX
  MOV BX,AX
  MOV AX,DI
  MUL CX
  MOV CX,AX
POP DX
POP AX
ADD DX,BX
ADD DX,CX

Comment fixer un nombre Single en Real sans coprocesseur mathématique

Étant sans doute un programmeur un peu fou disons de façon un peu plus poli ne voulant pas être à la merci d'une puce électronique, j'ai rencontrer, à certain moment donnée, en voulant charger le contenu d'une image de Studio 3D en mémoire, des problèmes de conversion de nombre de format Single à Real sans passer par un coprocesseur mathématique lequel n'étant pas disponible sur l'une de mes deux machines, c'est-à-dire celle posséder un excellant microprocesseur 386AMD-40Mhz. Sachant très bien que ce genre de situation a pu vous arrivez, car contrairement à ce que les fabricants croient, nous ne fonctionnons pas tous sur des Pentium INTEL. Au bout de multiples péripéties ayant durant environ 5 heures, j'ai résolu le problème en utilisant une technique assez simple de manipulation des octets de façon individuel. Après se calvaire, ayant le sang bouillant de vitriol et comprenant presque les paroles de la musique allemande de Rammstein, j'ai décidée de ne pas garder secret se genre de formule exaigu. L'idée réside essentiellement dans le fait de déplacer les bits au bonne endroit en tenant compte des limites du format. En voici d'ailleurs le code en langage de programmation Turbo Pascal :

Function SingleToReal(X:Single):Real;
Var
   SingleByte:Record
      A,B,C,D:Byte;
   End Absolute X;
   Y:Real;
   RealByte:Record
      A,B,C,D,E,F:Byte;
   End Absolute Y;
Begin
   Y:=0.0;
   RealByte.A:=(SingleByte.D shl 1)+1+((SingleByte.C and$80)shr 6);
   If(RealByte.A and 1=1)and(SingleByte.D and$F<>$F)Then Inc(RealByte.A);
   RealByte.D:=SingleByte.A;
   RealByte.E:=SingleByte.B;
   RealByte.F:=(SingleByte.C and$7F)or(SingleByte.D and$80);
   SingleToReal:=Y;
End;

Si vous n'arrivez pas a une réponse parfaitement exacte, vous n'avez qu'a utiliser des constantes en fonction de l'octet SingleByte.D envers RealByte.A, toutefois la précision a toujours été suffisante pour mes besoins logiciel, j'imagine que pour les vôtres aussi, en tout cas cela vous donnera une piste intéressante afin que vous compreniez le problème.

Comment fixer un nombre Double en Real sans coprocesseur mathématique

Plus tard, m'ayant frotter, lorsque de je me suis frotter un l'écriture d'un tableur supportant le format de Lotus 1-2-3, je dus me rendre compte qu'un problème des plus choquant m'attendait à nouveau, celui de devoir transformer leur format Double (un format à virgule flottante) en format Real compatible avec le compilateur Turbo Pascal . Bien qu'il soit possible à l'aide d'un 80x87 avec se compilateur de transformer ses nombres d'un format à l'autre, il est cependant impossible de le faire sans coprocesseur mathématique. Je ne pouvais me résoudre à l'idée d'être à la merci d'une puce électronique! J'ai donc du élaborer des routines permettant de surmonter cette situation. L'idée réside essentiellement dans le fait de déplacer les bits au bonne endroit en tenant compte des limites du format. Voici donc le code en langage de programmation Turbo Pascal :

Type
   TPReal=Record
      Exponent:Byte;
      Mantissa1:Word;
      Mantissa2:Word;
      MS:Byte;
   End;
   DoubleStruct=Record
      Mantissa1:Word;
      Mantissa2:Word;
      Mantissa3:Word;
      M4ES:Word;
   End;
{ Cette fonction permet de convertir un nombre de format «Double» en format «Real». }
Function DoubleToReal(X:Double):Real;
Var
   DStruct:DoubleStruct Absolute X;
   Real48:TPReal;
   Y:Real Absolute Real48;
Begin
   FillChar(Real48,SizeOf(Real48),0);
   Real48.Exponent:=((DStruct.M4ES and$7FF0)shr 4)-894;
   Real48.MS:=(((((DStruct.M4ES and$000F)shl 3)or
                        ((DStruct.Mantissa3 and$E000)shr 13)))and$7F)or(Hi(DStruct.M4ES)and$80);
   Real48.Mantissa2:=(((DStruct.Mantissa3 and$1FFF)shl 3)or((DStruct.Mantissa2 and$E000)shr 13));
   Real48.Mantissa1:=(((DStruct.Mantissa2 and$1FFF)shl 3)or((DStruct.Mantissa1 and$E000)shr 13));
   DoubleToReal:=Y;
End;
{ Cette fonction permet de convertir un nombre de format «Real» en format «Double».}
Procedure RealToDouble(X:Real;Var Y:Double);
Var
   DStruct:DoubleStruct Absolute Y;
   Real48:TPReal Absolute X;
Begin
   DStruct.M4ES:=((Real48.MS and$80)shl 8);
   DStruct.M4ES:=DStruct.M4ES or((Real48.Exponent+894)shl 4);
   DStruct.M4ES:=DStruct.M4ES or((Real48.MS and$7F)shr 3);
   DStruct.Mantissa3:=((Real48.MS and$7F)shl 13)or(Real48.Mantissa2 shr 3);
   DStruct.Mantissa2:=(Real48.Mantissa2 shl 13)or(Real48.Mantissa1 shr 3);
   DStruct.Mantissa1:=Real48.Mantissa1 shl 13;
End;

Si vous n'arrivez pas a une réponse satisfaisante, il vous restera à vous tourner vers le code écrit en langage de programmation C/C++.

Déterminer l'identité de deux matrices

La procédure suivante écrite en langage de programmation Turbo Pascal permet de déterminer l'identité de deux matrices, il restera cependant à définir un type Matrice pour un tableau de deux dimensions avec des variables réels:

Procedure IdentiteMatrice(Var R:Matrice);
Var I,J:Word;
Begin
   For J:=1to(DimensionY)do For I:=1to(DimensionX)do If(I=J)Then R[I,J]:=1.0 Else R[I,J]:=0.0;
End;

Additionner deux matrices

Cette procédure écrite en langage de programmation Turbo Pascal permet d'additionner le contenu de deux matrices, toutefois il restera à définir un type Matrice pour un tableau de deux dimensions avec des variables réels:

Procedure AdditionneMatrice(Const M,N:Matrice;Var R:Matrice);
Var I,J:Word;
Begin
   For J:=1to(DimensionY)do For I:=1to(DimensionX)do R[I,J]:=M[I,J]+N[I,J];
End;

Le scalaire de deux matrices

Cette procédure écrite en langage de programmation Turbo Pascal permet d'effectuer le produit scalaire de deux matrices, toutefois il restera à définir un type Matrice pour un tableau de deux dimensions avec des variables réels:

Procedure ScalaireMatrice(L:Real;Const M:Matrice;Var R:Matrice);
Var I,J:Word;
Begin
   For J:=1to(DimensionY)do For I:=1to(DimensionX)do R[I,J]:=L*M[I,J];
End;

La multiplication de deux matrices

Cette procédure écrite en langage de programmation Turbo Pascal permet d'effectuer la multiplication de deux matrices, toutefois il restera à définir un type Matrice pour un tableau de deux dimensions avec des variables réels:

Procedure MultiplieMatrice(Const M,N:Matrice;Var R:Matrice);
Var I,J,K:Word;
Begin
   For I:=1to(Dimension)do For J:=1to(Dimension)do Begin
      R[I,J]:=0;
      For K:=1to(Dimension)do R[I,J]:=R[I,J]+M[I,K]*N[K,J];
   End;
End;

Calcul à virgule fixe

A la fin des années 1980, on assista à l'arriver des VGA et des démos des Hackers. Ceux-ci avait l'extraordinaire talent d'avoir des performances phénoménal avec un équipement pourtant lent et assez souvent désuet. L'un de leur secret, fut d'utiliser les nombres a virgule fixe plutôt que l'exploitation des virgules flottantes. On renoncera à la précision d'un nombre afin d'atteindre cette objectif. Généralement, on utilisera une structure de deux entiers. La première contiendra la partie entière se trouvant avant la virgule, tandis que l'autre contiendra la partie décimal. Prenons par exemple le nombre 18,2, bien la première partie contiendra 18 et l'autre contiendra la décimale 02 (à cause du facteur décimal 100). Voici une exemple:

Program VirguleFixe;
Type Fixe=Record {Structure d'un nombre à virgule fixe}
   PEntier:Integer;
   PDecimal:Integer;
End;

Var {Les variables temporaire pour cette exemple }
   Var1:Fixe;
   Var2:Fixe;
Const
   FactorDecimal=100; {Garde deux chiffres après la virgule }
   NombreDecimal=2;
{transforme un nombre à virgule fixe en chaîne de chiffres}
Function Fixe2String(FNombre:Fixe):String;
Var
   PDecimalStr:String; {Chaîne de caractères de la partie décimale}
   PEntierStr:String; {Chaîne de caractères de la partie entière}
   I:Word;
Begin
   If FNombre.PDecimal<0Then FNombre.PDecimal:=-FNombre.PDecimal; {Élimine le signe après la virgule}
   Str(FNombre.PDecimal:NombreDecimal,PDecimalStr);{crée la chaîne de caractères décimale (après la virgule)}
   For I:=0to(NombreDecimal)do If PDecimalStr[I]=' 'Then PDecimalStr[I]:='0'; {remplace les espaces par un '0'}
   Str(FNombre.PEntier,PEntierStr);
   {crée la chaîne de caractères de la partie entière (avant la virgule)}
   Fixe2String:=PEntierStr+','+PDecimalStr; {Concataine les deux chaîne de caractères}
End;
{Cette procédure convertit un nombre Real RNombre en nombre à virgule fixe FNombre }
Procedure Real2Fixe(RNombre:Real;Var FNombre:Fixe);Begin
   FNombre.PEntier:=Trunc(RNombre); {Partie entière}
   FNombre.PDecimal:=Trunc(Round(Frac(RNombre)*FactorDecimal)); {Partie décimale stockée comme entier }
End;
{Cette procédure remet le nombre à virgule fixe transmis en format légal}
Procedure Adjust(Var FNombre:Fixe);Begin
   If FNombre.PDecimal>FactorDecimal Then Begin
      Dec(FNombre.PDecimal,FactorDecimal); {si la partie décimale est trop grande }
      Inc(FNombre.PEntier); {la diminuer et augmenter la partie entière }
   End;
   If FNombre.PDecimal<-FactorDecimal Then Begin
      Inc(FNombre.PDecimal,FactorDecimal); {si la partie décimale est trop petite}
      Dec(FNombre.PEntier); {l'augmenter et diminuer la partie entière }
   End;
End;
{Cette procédure additionne FNombre1 à FNombre2 et par la suite mémorise le résultat dans Somme}
Procedure Add(Var Somme:Fixe;FNombre1,FNombre2:Fixe);
Var Resultat:Fixe;
Begin
   Resultat.PDecimal:=FNombre1.PDecimal+FNombre2.PDecimal; {Somme les parties décimales}
   Resultat.PEntier:=FNombre1.PEntier+FNombre2.PEntier; {Somme les parties entières}
   Adjust(Resultat); {Ajuste le résultat dans un format correct }
   Somme:=Resultat;
End;

{Soustrait FNombre2 de FNombre1 et mémorise le résultat dans la variable Difference }
Procedure Sub(Var Difference:Fixe;FNombre1,FNombre2:Fixe);
Var Resultat:Fixe;
Begin
   Resultat.PDecimal:=FNombre1.PDecimal-FNombre2.PDecimal; {Soustraction des parties décimales }
   Resultat.PEntier:=FNombre1.PEntier-FNombre2.PEntier; {Soustraction des parties entières }
   Adjust(Resultat); {Ajuste le résultat dans une forme correcte }
   Difference:=Resultat;
End;
{Cette procédure multiplie FNombre1 par FNombre et après mémorise le résultat dans la variable Produit }
Procedure Mul(Var Produit:Fixe;FNombre1,FNombre2:Fixe);
Var Resultat:LongInt;
Begin
   Resultat:=Var1.PEntier*FactorDecimal+Var1.PDecimal; {Multiplicande}
   Resultat:=Resultat*(Var2.PEntier*FactorDecimal+Var2.PDecimal); {Multiplicateur}
   Resultat:=Resultat div FactorDecimal; {Redivise par le facteur décimal de la variable FactorDecimal }
   Produit.PEntier:=Resultat div FactorDecimal;
   Produit.PDecimal:=Resultat mod FactorDecimal; {Calcul les parties entière et décimale }
End;
{Cette procédure divise FNombre1 par FNombre2 et ensuite mémorise le résultat dans la variable Quotient }
Procedure Divi(Var Quotient:Fixe;FNombre1,FNombre2:Fixe);
Var Resultat:LongInt; {Résultat intermédiaire}
Begin
   Resultat:=FNombre1.PEntier*FactorDecimal+FNombre1.PDecimal;
   {Dividende... }
   Resultat:=Resultat*FactorDecimal div (FNombre2.PEntier*FactorDecimal+FNombre2.PDecimal);
   {... multiplié par le facteur décimal pour plus de précision et divisé par le diviseur }
   Quotient.PEntier:=Resultat div FactorDecimal;
   {Extrait les parties entière et décimale }
   Quotient.PDecimal:=Resultat mod FactorDecimal;
End;

BEGIN
   WriteLn;
   Real2Fixe(-81.2,Var1
); {On a besoin de deux nombres pour la démonstration) }
   Real2Fixe(73.1,Var2);
   {Effectue un calcul avec chaque opérateur pour le tester:}
   Write(Fixe2String(Var1),'*',Fixe2String(Var2),'= ');
   Mul(Var1,Var1,Var2);
   WriteLn(Fixe2String(Var1));
   Write(Fixe2String(Var1),'÷',Fixe2String(Var2),'= ');
   Divi(Var1,Var1,Var2);
   WriteLn(Fixe2String(Var1));
   Write(Fixe2String(Var1),'+',Fixe2String(Var2),'= ');
   Add(Var1,Var1,Var2);
   WriteLn(Fixe2String(Var1));
   Write(Fixe2String(Var1),'-',Fixe2String(Var2),'= ');
   Sub(Var1,Var1,Var2);
   WriteLn(Fixe2String(Var1));

END.


PARTAGER CETTE PAGE SUR
Dernière mise à jour : Dimanche, le 1 mai 2016