CVF PG
CVF PG
CVF PG
August, 2001 Visual Fortran Version 6.6 Microsoft Windows 98, Windows Me, Windows 95, Windows 2000, or Windows NT Version 4
Copyright Information
Page 1 of 2
Copyright Information
Confidential computer software. Valid license from Compaq required for possession, use or copying. Consistent with FAR 12.211 and 12.212, Commercial Computer Software, Computer Software Documentation, and Technical Data for Commercial Items are licensed to the U.S. Government under vendor's standard commercial license. Compaq shall not be liable for technical or editorial errors or omissions contained herein. The information in this document is provided "as is" without warranty of any kind and is subject to change without notice. The warranties for Compaq products are set forth in the express limited warranty statements accompanying such products. Nothing herein should be construed as constituting an additional warranty. 2001 Compaq Computer Corporation Visual Fortran Home Page, Photographic images: Copyright 1997 PhotoDisc, Inc. Visual Fortran Home Page, Image: CERN, European Laboratory for Particle Physics: ALICE detector on CERN's future accelerator, the LHC, Large Hadron Collider.
Compaq, the COMPAQ logo, DEC, DEC Fortran, DIGITAL, OpenVMS, Tru64, VAX, VAX FORTRAN, and VMS are trademarks of Compaq Information Technologies Group, L.P. in the United States and other countries. ActiveX, Microsoft, Microsoft Press, MS-DOS, PowerPoint, Visual Basic, Visual C++, Visual J++, Visual Studio, Win32, Win32s, Windows, and Windows NT are trademarks of Microsoft Corporation in the United States and other countries. Intel and Pentium are trademarks of Intel Corporation in the United States and other countries. AMD and Athlon are trademarks of Advanced Micro Devices, Inc. Borland and Delphi are trademarks of Borland Software Corporation. CRAY is a registered trademark of Cray Research, Inc. IBM is a registered trademark of International Business Machines, Inc. IEEE is a registered trademark of the Institute of Electrical and Electronics Engineers,
Copyright Information
Page 2 of 2
Inc. IMSL and Visual Numerics are registered trademarks of Visual Numerics, Inc. Linux is a registered trademark of Linus Torvalds. OpenGL is a registered trademark of Silicon Graphics, Inc. OpenMP and the OpenMP logo are trademarks of OpenMP Architecture Review Board. Sun Microsystems is a registered trademark and Java is a trademark of Sun Microsystems, Inc. UNIX is a trademark of the Open Group in the United States and other countries. All other product names mentioned herein may be trademarks of their respective companies.
Page 1 of 2
Support has been added for 8-byte integer and 8-byte logical intrinsic data types. The Windows API interface header files DFWINTY.F90 and DFWIN.F90 (and its component files) have been extensively revised to work in both a 32-bit and a 64-bit environment. A new version of the Compaq Extended Math Library (CXML) is included with Visual Fortran Version 6.6. CXML Version 5.0 includes improved performance for many CXML routines, a new set of routines for solving sparse matrix problems using direct methods, sample files that show how to invoke the direct sparse solver routines from Fortran and C, changes to certain sparse iterative solver functions, changes in how messages can be printed with the iterative solver functions (including changes to the argument iounit), and other changes. The skyline solver routines are no longer supported. The auxiliary LAPACK routine XLAENV is no longer supplied by CXML; instead, the LAPACK routines DLAMCH and SLAMCH have been modified to return compile-time constants instead of computing values on each call. The types of Visual Fortran run-time libraries displayed in the visual development environment Fortran tab (Project menu, Settings dialog box) more closely match the run-time libraries available in Visual C++ (C/C++ tab). Also, the types of run-time libraries linked against while linking against debug versionsof DLL libraries have changed, as described in Visual Fortran/Visual C++ Mixed-Language Programs and Specifying Consistent Library Types. The following new graphics functions let you draw Bezier curves (based on fitting a cubic curve to four points):
l l
/annotations now provides information about optimizations in a listing file. /architecture and /tune options now support the pn4 keyword for Pentium 4 systems. In addition, the pn1, pn2, and pn3 keywords have been added (pn is an abbreviation for Pentium) as the preferred
Page 2 of 2
The /ccdefault:default option now allows other options, such as /vms, to set the default carriage control. The keyword /fpscomp:ldio_spacing, for list-directed output, controls whether a blank is inserted at run-time after a numeric value before a character value. This option is set by /fpscomp:general. /integer_size now supports a value of 64 to allow INTEGER and LOGICAL declarations to be treated as 8 bytes long (KIND=8). Specifying the /stand option (without keywords) now sets /stand:f95 (Fortran 95 standards checking) instead of /stand:f90 (Fortran 90 standards checking). If you specify /fast with /stand, /align:dcommons and /align:sequence (usually set by /fast) are not set.
A derived-type variable that is data initialized via default initialization of any of its components will no longer be saved by default. A RECORD variable that is data initialized via default initialization specified in its STRUCTURE declaration will no longer be saved by default. The Fortran preprocessor fpp supports the macros __DATE__ and __FILE__. When you request that fpp expand macros using "-fpp:-m", these expand into character literals that give the name of the source file being processed and the current date/time respectively. For additional new features and details, see the online release notes (relnotes.txt or relnotes.htm). New or significantly revised sections in the Programmer's Guide include Copying Projects and Viewing Fortran Data Types in the Debugger.
Beginning with Version 6.6, Visual Fortran no longer supports Windows NT Alpha systems. Changes to Compaq Array Visualizer are described separately in the Array Visualizer HTML Help documentation.
New Features for Compaq Visual Fortran Version 6.5 and.. Page 1 of 4
New Features for Compaq Visual Fortran Version 6.5 and 6.5A
New features added to Compaq Visual Fortran (Visual Fortran) Version 6.5A (since Visual Fortran Version 6.5) include the following:
l
The Fortran COM Server now allows you to select an out-of-process (.EXE) COM server. Users now have a choice of the type of server, either a DLL (in-process) COM server and an EXE (out-of-process) COM server. Other Fortran COM Server enhancements allow array arguments to be assumed shape and you can now add the ISupportErrorInfo interface to your server. The following new compiler options have been added:
l
/assume:[no]protect_constants specifies whether constant actual arguments can be changed. /check:arg_temp_created requests that a run-time informational message appear if actual arguments are copied into temporary storage before routine calls. /warn:ignore_loc requests that the compiler issues warnings when % LOC is stripped from an argument
The following new ATTRIBUTES directives have been added: ALLOW_NULL, DECORATE, IGNORE_LOC, and NOMIXED_STR_LEN_ARG The optional KIND argument has been added to INDEX, LEN_TRIM, MAXLOC, MINLOC, SCAN, and VERIFY intrinsic procedures. For additional new features and details, see the online release notes (relnotes.txt or relnotes.htm). New or significantly revised sections in the Programmer's Guide based on customer feedback include Coding Requirements for Fortran Windows Applications, Using Menus and Dialogs in SDI and MDI Fortran Windows Applications, Using the Resource Editor to Design a Dialog, and Compatibility with Compaq Fortran on Other Platforms. The section Pointer Arguments in the Language Reference has been significantly revised based on customer feedback.
New features added to Compaq Visual Fortran Version 6.5 (since Visual Fortran Version 6.1) include the following:
New Features for Compaq Visual Fortran Version 6.5 and.. Page 2 of 4
The Fortran COM Server project type and the Fortran COM Server Wizard, described in Creating a COM Server. The Fortran COM Server Wizard helps you create a COM server or dual interface server. In addition, new information about using COM and Automation is provided in Getting a Pointer to an Object's Interface. New COM routines are provided:
l
COMStringFromGUID passes a GUID and returns the corresponding string representation. COMIsEqualGUID determines if two GUIDs are the same.
l l
Visual Fortran now generates optimized code for the IntelTM PentiumTM III, AMDTM K6, and AMD AthlonTM architectures, by providing new keywords for the /arch and /tune compiler options. New intrinsic procedures added:
l l
INT_PTR_KIND returns the INTEGER KIND that will hold an address. KIND= is now an optional argument to the intrinsic procedures LEN, SHAPE, SIZE, LBOUND, and UBOUND. For x86 systems, POPCNT, POPPAR, LEADZ, and TRAILZ perform bitrelated operations on integer data.
CLEARSTATUSFPQQ exception flags in the floating-point processor status word (x86 processors). GETEXCEPTIONPTRSQQ returns a pointer to C run-time exception information pointers appropriate for use in signal handlers established with SIGNALQQ or direct calls to the C rtl signal() routine (x86 processors). For additional information about GETEXCEPTIONPTRSQQ, see Obtaining Traceback Information with TRACEBACKQQ and TRACEBACKQQ.
The new QuickWin graphics routine SETTEXTCURSOR has been added to set the height and width of the text cursor (the caret) for the window in focus. The interface to the AUTOAddArg subroutine has changed. The output_arg argument (LOGICAL) is now the intent_arg argument (INTEGER). New code should use the new documented interface. The old interface is supported for compatibility purposes. New information about advanced exception handling and setting up various
New Features for Compaq Visual Fortran Version 6.5 and.. Page 3 of 4
types of handlers for x86 systems is provided in Advanced Exception and Termination Handling Considerations.
l
For non-native unformatted files, you can now use the FORT_CONVERT.ext or FORT_CONVERT_ext environment variable method to specify that files with certain file extensions (such as .DAT) are in a specified non-native format (see Methods of Specifying the Data Format). (Visual Fortran's native numeric formats are little endian, including IEEE floating-point formats.) Previous versions of Visual Fortran allowed only procedures and COMMON blocks to have the DLLEXPORT or DLLIMPORT ATTRIBUTE. You can now export/import module variables and arrays. A Visual Fortran compiled module now contains all of the information from modules used (USE statement) by the module. This may greatly increase the size of compiled modules, especially if the modules contain a USE DFWIN or USE DFWINTY statement. There are ways to minimize the information contained in compiled modules when calling Win32 routines, as described in Calling Win32 Routines and the Visual Fortran Windows Module. In order to conform with clarified wording in the Fortran 95 standard, the compiler has been changed so that when a READ statement encounters an end-of-file condition, and there is no END specifier but there is an ERR specifier, the ERR= branch is not taken. Similarly, if an end-of-record condition occurs but there is no EOR specifier, an ERR branch is not taken. If you do not specify a routine to handle such errors with the IOSTAT specifier, omitting the END or EOR specifier results in a severe run-time error (such as numbers 24 or 268 respectively). The run-time system has been changed to perform more thorough edit checking on list-directed input. In accordance with the Fortran 95 Standard, the run-time system no longer accepts as numeric input "+", "-", ".", "D", "E", or "Q" without expressing at least 1 digit. For example, the run-time system used to allow a single "+" to convert to a 0, but now it will return a FOR$IOS_LISIO_SYN (number 59) error. In addition, ambiguous expressions such as "+-" and "--" will be rejected Support has been added for reading nondelimited character strings as input for character NAMELIST items. The %VAL and %REF used on actual arguments now override any argument-passing mechanism specified in an explicit interface. New Visual Fortran Samples (described in Roadmap to the Visual Fortran Samples) are provided, including:
l
New Features for Compaq Visual Fortran Version 6.5 and.. Page 4 of 4
l l
DIALOG sample: Celsicon ExceptionHandling samples: ClearFP, Cslexp2, Cslexp4, GetEptrs, Vbvf1, and Winexcp1 Isovar sample (ISO varying strings) Mixed-language samples, new samples for MASM and Delphi QuickWin samples: Conapp, Dirkeys4, PostMini, and Resize Isovar, an ISO varying string sample
l l l l l l l
Compiling modules is now faster. Concurrent-use licensing is now available. For additional new features and details, see the online release notes (relnotes.txt or relnotes.htm), located in the root directory of the Visual Fortran CD-ROM or installed in Program Files\Microsoft Visual Studio\Df98. New or significantly revised sections in the Programmer's Guide based on customer feedback include Specifying Consistent Library Types and Using the Console.
Changes to Compaq Array Visualizer are described separately in the Array Visualizer HTML Help documentation.
Page 1 of 6
The Compaq Extended Math Library (CXML) is now provided with the Visual Fortran kit (all editions). CXML is a highly efficient set of math library routines that covers Basic Linear Algebra (BLAS), Linear Algebra Routines (LAPACK), sparse linear system solvers, sorting routines, random number generation, and signal processing functions. In addition to being provided with Visual Fortran, CXML is also available on other Compaq Fortran platforms. CXML is:
l
Provided with Compaq Fortran Version 5.3 (or later) for Tru64 UNIX Alpha systems and Compaq Fortran Version 1.0 (or later) for Linux Alpha Systems Provided with Compaq Visual Fortran Version 6.1 (or later) Provided with Compaq Fortran Version 7.3 (or later) for OpenVMS Alpha systems Available as a separate download for most platforms (see the CXML home page at http://www.compaq.com/math)
l l
For more information on CXML, see Using the Compaq Extended Math Library (CXML) in the Programmer's Guide or the online PDF file Cxmlref.pdf. For information on online PDF files, see The IMSL Routines and CXML Routines Online PDF Documentation.
l
IMSL Version 4 libraries are now provided with the Professional Edition (prior to Visual Fortran 6.1, Visual Fortran provided IMSL Version 3 libraries). The IMSL routines reference documentation is available in PDF format (not HLP format). For information on using the IMSL libraries, see Using the IMSL Mathematical and Statistical Libraries in the Programmer's Guide. For information on online PDF files, see The IMSL Routines and CXML Routines Online PDF Documentation. For more information about IMSL and Visual Numerics, Inc., see http://www.vni.com.
You can use ActiveX Controls in your dialog boxes. See Using ActiveX Controls in the Using Dialogs section in the Programmer's Guide. Edit Box control enhancements:
Page 2 of 6
You can now use multi-line edit controls and the maximum size restriction of 256 characters has been removed. The Edit Box control supports 2 additional integer indexes (see Using Edit Boxes):
l
DLG_TEXTLENGTH - Sets or Gets the current length of the text in the edit box. DLG_POSITION - Sets or Gets the current text cursor position in the edit box.
The dialog box supports a new callback index, DLG_SIZECHANGE (see Control Indexes). This callback is invoked when the size of the dialog box changes. New dialog routines are provided:
l l
DlgSetTitle - Sets the title of the dialog box. DlgFlush - Updates the display of the dialog box. Useful when changing the dialog box outside of a dialog callback. DlgModalWithParent - Allows you to specify the parent window of a modal dialog box. Useful in an SDI or MDI Windows application that uses modal dialog boxes. DlgIsDlgMessageWithDlg - Allows you to specify which dialog box to check. Useful when the dialog box was created in a DLL.
New SPORT_xxx (Serial Port) routines have been added. The SPORT_xxx collection of routines helps Fortran programmers do basic input and output to serial communication ports. The programming model is much the same as a normal file except the user does a connect/release instead of an open/close. Two types of read and write operations are provided:
l l
Read and write arbitrary data from/to the port Read and write line terminated data
Calls are provided to set basic port parameters such as baud rate, stop bits, timeouts, and so on. Additionally, a call is provided to return the WIN32 handle to the port so that raw WIN32 APIs may be used to implement additional needs of the programmer. The documentation overview can be found in Using the Serial I/O Port Routines in the Programmers Guide. For a detailed description of the routines, see the Language Reference, A to Z Reference under S (SPORT_xxx), such as SPORT_CONNECT.
Page 3 of 6
The new subroutine TRACEBACKQQ allows you to initiate a stack trace in your program whenever you like. The input arguments allow you to specify your own message text to head the output and other options as well. See the following for more information:
l
Obtaining Traceback Information with TRACEBACKQQ in the Programmer's Guide TRACEBACKQQ in the Language Reference, A to Z Reference.
l l
In Developer Studio, you can now print a Fortran source file in color (requires a color printer). After you open the source file in the text editor, in the File menu select Print Colorized Fortran. This has been tested with the Internet Explorer and Netscape browsers. Because Print Colorized Fortran does not know when printing has completed, it creates and leaves the HTML file in your TEMP directory. Delete previously printed HTML files from your TEMP directory to free up disk space. Print Colorized Fortran uses the FSC utility to create an HTML file with the coloring information and then instructs the application on your system that handles HTML files to print the file. If you encounter a problem using the Print Colorized Fortran item, please use the FSC utility directly. The FSC.EXE utility takes an .F90 or .FOR source file and produces an HTML version that is colorized per the current Developer Studio Color settings in the registry. It has default colors that are used if the current registry settings are not available. The resultant HTML file may be printed or viewed with your favorite browser. This utility may be invoked from a command prompt (use FSC -? or FSC /h for help) or from within Developer Studio.
VFRUN provides a self-extracting executable file that installs redistributable Visual Fortran run-time components on the target system. These run-time components include DLLs and other files, as described in "Redistributing Visual Fortran Files" in Compaq Visual Fortran Installing and Getting Started. VFRUN files are provided for different Visual Fortran releases and for x86 and Alpha systems. For example, for Visual Fortran v6.1, VFRUN61i.exe is provided for x86 systems and VFRUN61a.exe is provided for Alpha systems. You can download VFRUN from the Visual Fortran Web site, http://www.compaq.com/fortran (click on Downloads and Updates, then Run-Time Redistributables Kit). Please download the VFRUN kit for the version of Visual Fortran used to create the executable application.
Visual Fortran (VF) Reporter is a new tool that helps you report problems by e-mail to the Visual Fortran team. This tool gathers and displays system information and guides you to send relevant information in an e-mail message. This tool can be optionally installed during Visual Fortran
Page 4 of 6
installation. For information on technical support and using VF Reporter, see Compaq Visual Fortran Installing and Getting Started.
l
Greater Access to Windows APIs Additional interface definitions have been provided to simplify calling Win32 routines from Visual Fortran:
l
DFWINTY.F90 has 980 new derived types and 4070 new Fortran parameter constants added. These new items represent the difference between Microsoft Visual C++ 4 header files and Visual C++ 6 header files. USER32.F90 has 72 new interfaces to routines in USER32.LIB. These new routine interfaces again represent the difference between Microsoft Visual C++ 4 header files and Visual C++ 6 header files. KERNEL32.f90 has 50 new interfaces to routines in KERNEL32.LIB. GDI32.F90 has 25 new interfaces to routines in GDI32.LIB. ADVAPI32.F90 has 50 new interfaces to routines in ADVAPI32.LIB.
l l l
In all these files, the new material is under conditional compilation so that defining "__DO_NOT_INCLUDE_VC6_ITEMS" will remove the new items. Search for the string above to locate the new items which will be found towards the ends of the files. The files are shipped without the string being defined, and the mod files will contain the new items.
l
The /align option now includes the /align:sequence keyword to allow alignment of derived type data with the SEQUENCE statement. This option appears in the Fortran Data category in the Project Settings dialog box (see Categories of Compiler Options). The /ccdefault:keyword option controls carriage control use when writing to a terminal screen. This new option appears in the Run-Time category in the Project Settings dialog box. The /cxml option requests that the Compaq Extended Math Library (CXML) library be linked against (provided for documentation purposes only). This new option appears in the Library category in the Project Settings dialog box. The /fast option now sets the /arch:host /tune:host /align:sequence options.
Page 5 of 6
The /imsl option ensures that the IMSL libraries will be passed to the linker (your program needs to specify the appropriate USE statements and set the IMSL environment variables). This new option appears in the Library category in the Project Settings dialog box.
Visual Fortran puts literals into read-only memory so storing into a dummy argument that has a constant as its associated actual argument will result in an access violation and program termination: call f(0) ... subroutine f(i) i=1
Two new Fortran source directives for controlling optimizations are now available:
l l
For an overview of these directives, see Compiler Directives Related to Performance in the Programmer's Guide. For more detail, see IVDEP Directive and UNROLL Directive in the Language Reference.
l
Version 6.1 for x86 (Intel) systems contains all of the applicable updated files from the Microsoft Visual Studio 6.0 Service Pack 3. Therefore, Version 6.1 x86 users do not need to install Visual Studio 6.0 Service Pack unless you have other Visual Studio products. Version 6.1 for Alpha systems contains all of the applicable updated files from the Microsoft Visual Studio 6.0 Service Pack 2. You can now maintain previously saved Fortran Environment Project Settings (see Saving and Using the Project Setting Environment for Different Projects. Use the Manage Saved Fortran Environment icon in the Fortran toolbar. Also, application wizards that assist in creating a new Fortran Windows project have been enhanced. HTML Help now supports the ability to define (View menu, Define Subset... item) and use subsets of the current collection of HTML Help titles (see Compaq Visual Fortran Installing and Getting Started). This is especially useful when using the full-text search capabilities of HTML Help. HTML help no longer includes embedded Adobe Acrobat PDF files. Instead, the PDF files ship on the Visual Fortran CD-ROM (and are installed if
Page 6 of 6
requested). For details, see The IMSL Routines and CXML Routines Online PDF Documentation. This reduces the size of the Visual Fortran HTML help (CHM) file and minimizes the possibility of software problems displaying embedded PDF files reported on certain systems.
l
For additional details, see the online release notes, installed in Files\Microsoft Visual Studio\Df98.
Program
Changes to Compaq Array Visualizer are described separately in the Array Visualizer HTML Help documentation.
Page 1 of 2
Building Programs and Libraries Using the Compiler and Linker from the Command Line Compiler and Linker Options Debugging Fortran Programs Performance: Making Programs Run Faster
Special coding and related considerations for certain Visual Fortran project types:
Topics about programming with Visual Fortran on Windows 2000, Windows NT 4.0, Windows Me, Windows 98, and Windows 95 systems:
Portability and Design Considerations Using Dialogs Drawing Graphics Elements Using Fonts from the Graphics Library Using National Language Support Routines Portability Library Files, Devices, and I/O Hardware Using COM and Automation Objects Creating a COM Server Programming with Mixed Languages Creating Multithread Applications
Page 2 of 2
The Floating-Point Environment Advanced Exception and Termination Handling Considerations Converting Unformatted Numeric Data Hexadecimal-Binary-Octal-Decimal Conversions
Using the IMSL Mathematical and Statistical Libraries Using the Compaq Extended Math Library (CXML)
Note: Visual Fortran contains many extensions to the full ANSI/ISO standard language. In this book, the Visual Fortran extensions to the Fortran 95 standard appear in this color.
Page 1 of 5
General Conventions The Programmer's Guide uses the following general conventions. (Note that in most cases, blanks are not significant in Fortran 90 or 95.) When You See This Extensions to Fortran 95 Here Is What It Means Dark teal type indicates extensions to the Fortran 95 Standard. These extensions may or may not be implemented by other compilers that conform to the language standard. Uppercase (capital) letters indicate filenames and MS-DOS-level commands used in the command console. Words in italics indicate placeholders for information that you must supply. A filename is an example of this kind of information. Italics are also used to introduce new terms. Items inside single square brackets are optional. In some examples, square brackets are used to show arrays. Braces and a vertical bar indicate a choice among two or more items. You must choose one of the items unless all of the items are also enclosed in square brackets.
expression
[optional item]
{choice1 | choice2}
Page 2 of 5
s[, s]...
A horizontal ellipsis (three dots) following an item indicates that the item preceding the ellipsis can be repeated. In code examples, a horizontal ellipsis means that not all of the statements are shown. This term refers to Windows options, OpenVMS qualifiers, and Tru64 UNIX and Linux options that can be used on the compiler command line. This kind of type is used for program examples, program output, and error messages within the text. An exclamation point marks the beginning of a comment in sample programs. Continuation lines are indicated by an ampersand ( & ) after the code at the end of a line to be continued and before the code on the following line. Bold capital letters indicate Compaq Fortran statements, functions, subroutines, and keywords. Keywords are a required part of statement syntax, unless enclosed in brackets as explained above. In the sentence, "The following steps occur when a DO WHILE statement is executed," the phrase DO WHILE is a Compaq Fortran keyword. Bold lowercase letters are used for keywords of other languages. In the sentence, "A Fortran subroutine is the equivalent of a function of type void in C," the word void is a keyword of C. This term refers to language information that is common to ANSI FORTRAN 77, ANSI/ISO Fortran 95 and 90, and Compaq Fortran (formerly DIGITAL Fortran).
compiler option
Fortran
Page 3 of 5
This term refers to language information that is common to ANSI/ISO Fortran 95 and ANSI/ISO Fortran 90. This term refers to language information that is common to ANSI/ISO Fortran 95. This term refers to language information that is common to ANSI/ISO Fortran 90. These terms refer to the same language. These terms refer to the same operating system. These terms refer to the same operating system. This term refers to all supported Microsoft Windows operating systems. (See labels WNT and W9* in Platform Labels below.) This term refers to the INTEGER(KIND=1), INTEGER(KIND=2), INTEGER (INTEGER (KIND=4)), and INTEGER(KIND=8) data types as a group. This term refers to the REAL (REAL (KIND=4)) and DOUBLE PRECISION (REAL (KIND=8)) data types as a group. This term refers to the COMPLEX (COMPLEX(KIND=4)), DOUBLE COMPLEX (COMPLEX(KIND=8)), and COMPLEX (KIND=16) data types as a group. This term refers to the LOGICAL(KIND=1), LOGICAL(KIND=2), LOGICAL (LOGICAL (KIND=4)), and LOGICAL(KIND=8) data types as a group.
Fortran 95 Fortran 90 Compaq Fortran, DIGITAL Fortran OpenVMS, VMS Tru64 UNIX, DIGITAL UNIX Windows systems
integer
real
complex
logical
Page 4 of 5
The Programmer's Guide uses certain conventions for language syntax. For example, consider the following syntax for the PARAMETER statement: PARAMETER [(] c = expr [, c = expr ]...[)] This syntax shows that to use this statement, you must specify the following:
l l l
The keyword PARAMETER An optional left parenthesis One or more c=expr items, where c is a named constant and expr is a value; note that a comma must appear between c=expr items. The three dots following the syntax means you can enter as many c=expr items as you like. An optional terminating right parenthesis
The dark teal brackets ([ ]) indicate that the optional parentheses are an extension to standard Fortran. Platform Labels A platform is a combination of operating system and central processing unit (CPU) that provides a distinct environment in which to use a product (in this case, a language). For example, Microsoft Windows 98 on x86 is a platform. Information applies to all supported platforms unless it is otherwise labeled for a specific platform (or platforms), as follows: VMS U*X TU*X Applies to OpenVMSTM on Alpha processors. Applies to Tru64 UNIX and Linux on Alpha processors. Applies to Tru64 UNIX on Alpha processors.
Page 5 of 5
Applies to Microsoft 2000 and Windows NT 4.0 operating systems on AMD and Intel x86 processors, and 64-bit Windows operating systems on Intel IA-64 processors. Applies to Microsoft Windows 98, Windows Millennium Edition (Me), and Windows 95 operating systems on AMD and Intel x86 processors. Applies to operating systems on Alpha processors. Applies to 32-bit Windows operating systems on AMD and Intel x86 processors (see "System Requirements and Optional Software" in Compaq Visual Fortran Installing and Getting Started). Applies to 64-bit Windows operating systems on IA-64 processors.
W9*
Alpha ia32
ia64
Page 1 of 58
l l l l l
Overview of Building Projects Types of Projects you can build Defining Your Project and selecting project features with the visual development environment Errors During the Build Process Compiler Limits Running Fortran Applications Copying Projects Visual Fortran Samples
l l
Building programs and libraries at the command line, see: Using the Compiler and Linker from the Command Line and Using Visual Fortran Tools. Using the Debugger, see Debugging Fortran Programs. Details about using the development environment, see the Visual C++ User's Guide.
Building Programs and Libraries consistent source code base from which to work.
Page 2 of 58
When you create a new project, the Microsoft visual development environment automatically creates Debug and Release configurations for you. The default configuration is the Debug configuration. To specify the current configuration, select Set Active Configuration from the Build menu. Projects are contained in a workspace. When you create a new project, you indicate whether the project is created in a new workspace or an existing workspace. To open an existing project, you open its workspace. A workspace can contain multiple projects. Once you open a workspace, the development environment displays a FileView pane, which displays the files contained in the project, and lets you examine visually the relationships among the files in your project. Modules, include files, or special libraries your program uses are automatically listed as dependencies. The output window displays information produced by the compiler, linker, Find in Files utility, and the profiler. You can specify build options in the Project menu Settings dialog box, for one of the following:
l l l
For example, you can specify certain kinds of compiler optimizations for your project in general, but turn them off for certain configurations or certain files. Once you have specified the files in your project, the configurations that your project is to build, and the tool settings for those configurations, you can build the project with the commands on the Build menu. In addition to using the visual development environment, you can also use the command line (DF command). You can also develop the application from the visual development environment and export a makefile for use in command-line processing (NMAKE command). Your choice of development environment determines what you can do at each stage. The following diagram illustrates the development process for using the visual development environment or command line: Example of Development Process
Page 3 of 58
Notes in the diagram point to places where you can read more about a particular part of the development process: 1. 2. 3. 4. 5. 6. 7. Building Programs and Libraries (the current chapter) Using the Compiler and Linker from the Command Line Using Visual Fortran Tools The Floating-Point Environment and Handling Run-Time Errors Debugging Fortran Programs Profiling Code from the Command Line Analyze Program Performance
For more overview information about building projects with the visual development environment, see:
l l l
Page 4 of 58
a folder is a visual representation of files in a project. Folders show the order in which Visual Fortran compiles the files, and the relationship of source files to their dependent files, such as modules. When you initially create a project, the Project Workspace window contains some default panes, accessible through tabs at the bottom of the window, to display information about the content of the project. You can also open an output window, which has panes that display build output, debug output, Find in Files output, and profiler output. In addition to the default panes, you can create customized panes to organize and display project information in ways most useful to you. You can access information about components of the project from the panes in the project window. Double-clicking any item in a pane displays that item in an appropriate way: source files in a text editor, dialog boxes in the dialog editor, help topics in the information window, and so on. Be sure to select the appropriate pane when using the menu commands, in particular the Save and Save As commands. Commands on the File menu affect only the window that currently has the focus.
Menu Options
Menu options that are available to you may look different, depending on which window or pane has current focus. The Debug menu, for example, is only visible when you are debugging. The visual development environment has the following menu bars and toolbars:
l l l l l l l l l l
Standard menu bar Standard toolbar Build toolbar Build minibar Resource toolbar Edit toolbar Debug toolbar Browse toolbar Fortran toolbar Controls toolbar
In the Tools menu, click Customize item Click the Toolbar tab Select (click) the toolbar name
Page 5 of 58
Move the mouse pointer into the project window and click the right mouse button.
You can now select project commands that are appropriate for your current selection in the project window.
Types of Projects
When you create the project, you must choose a project type. You need to create a project for each binary executable file to be created. For example, the main Fortran program and a Fortran dynamic-link library (DLL) would each reside in the same workspace as separate projects. The project type specifies what to generate and determines some of the options that the visual development environment sets by default for the project. It determines, for instance, the options that the compiler uses to compile the source files, the static libraries that the linker uses to build the project, the default locations for output files, defined constants, and so on. You can build seven kinds of projects with Visual Fortran. You specify the project type when you create a new project. They are summarized in the following table: Project Type Fortran Console Application (.EXE) Key Features Single window main projects without graphics (resembles character-cell applications). Requires no special programming expertise. For a Visual Fortran Sample of a Console Application, see ...\COMMON\MSdev98\MYPROJECTS\CELSIUS, as described in "Opening an Existing Project" in Compaq Visual Fortran Installing and Getting Started.
Building Programs and Libraries Fortran Standard Graphics Application (.EXE) Fortran QuickWin Application (.EXE) Fortran Windows Application (.EXE) Fortran COM Server (.DLL)
Page 6 of 58
Single window main projects with graphics. The programming complexity is simple to moderate, depending on the graphics and user interaction used. Samples of Standard Graphics Applications (QuickWin single window) resemble those for QuickWin Applications (see below). Multiple window main projects with graphics. The programming complexity is simple to moderate, depending on the graphics and user interaction used. Samples of QuickWin Applications (QuickWin multiple window) are in ...\DF98 \SAMPLES\QUICKWIN, such as QWPIANO and QWPAINT. Multiple window main projects with full graphical interface and access to all Win32 API routines. Requires advanced programming expertise and knowledge of the Win32 routines. Samples of Win32 Applications are in ...\DF98 \SAMPLES\ADVANCED\Win32, such as PLATFORM or POLYDRAW. Uses the Component Object Model (COM) to implement a COM server. COM supports client-server interaction between a user of an object, the client, and the implementor of the object, the server. Clients may be written in Visual Fortran using the Fortran Module Wizard or in other languages, such as Visual C++ and Visual Basic. Requires advanced knowledge of COM. Samples of Fortran COM Server (and COM client) applications are in ...\DF98\SAMPLES\ADVANCED\COM, such as Adder. Library routines to link into .EXE files. Library routines to associate during execution.
The first four projects listed in the preceding table are main project types, requiring main programs. The last two are library projects, without main programs. The project types are discussed in detail in:
l l l l l l
Console Application Standard Graphics Application QuickWin Application Windows Application COM Server Static library
Page 7 of 58
When migrating legacy applications, choosing a project type that will minimize porting effort depends on what the application does. A character-cell application that does not use any graphics (such as a program ported from a UNIX system) can usually be converted into a Fortran Console Application. When a legacy Windows 3.1 (or older) application uses graphics calls, you can try a Fortran Standard Graphics Application or a Fortran QuickWin Application project type. However, be aware that with such legacy graphics applications:
l
The routine names of certain legacy graphic QuickWin routines might need to be changed to the current routine names (and their current argument lists). For example, the following table lists some of known obsolete routines and their supported replacement routines: Obsolete QuickWin Routine Replacement Routine SETGTEXTVECTOR GETGTEXTVECTOR GETHANDLEQQ WGGETACTIVE WGSETACTIVE INITIALMENU SETTEXTROWS SETGTEXTROTATION GETGTEXTROTATION GETHWNDQQ GETACTIVEQQ SETACTIVEQQ INITIALSETTINGS SETWINDOWCONFIG
The character-oriented method of user interaction can be modernized with menus, mouse clicks, and dialog boxes. If the routine uses the PEEKCHARQQ routine, see the PEEKAPP or PEEKAPP3 Visual Fortran Samples in the QuickWin folder.
After you select your project type, you need to define your project (see Defining Your Project). If you need to use the command line to build your project, you can:
l
Initially use the visual development environment and later create a makefile (see The Project Makefile) Use the DF command compiler options to specify the project type (see Specifying Project Types with DF Command Options) Create the application from the DF command line (see Using the Compiler and Linker from the Command Line)
Page 8 of 58
A Fortran Console application (.EXE) is a character-based Visual Fortran program that does not require screen graphics output. It looks similar to a program running on a UNIX workstation or a terminal connected to a mainframe computer. Fortran Console projects operate in a single window, and let you interact with your program through normal read and write commands. Console applications are better suited to problems that require pure numerical processing rather than graphical output or a graphical user interface. This type of application is also more transportable to other platforms than the other types of application. Fortran Console applications can be faster than Fortran Standard Graphics or Fortran QuickWin graphics applications, because of the resources required to display graphical output (see Using the Console). Any graphics routine that your program calls will produce no output, but will return error codes. A program will not automatically exit if such an error occurs, so your code should be written to handle this condition. With a Fortran Console project, you can use static libraries, DLLs, and dialog boxes, but you cannot use the QuickWin functions. You can select the multithreaded libraries with this and all of the other Fortran project types. As with all Windows command consoles, you can toggle between viewing the console in a window or in full-screen mode by using the ALT+ENTER key combination. A sample Fortran Console project is Celsius (see \MYPROJECTS\CELSIUS), as described in "Opening an Existing Project" in Compaq Visual Fortran Installing and Getting Started or Defining Your Project.
Page 9 of 58
APIs for loading and unloading bitmap files. Standard graphics applications should be written as multithreaded applications. (For information about multithreaded programs, see Creating Multithread Applications.) Fortran standard graphics (QuickWin single document) applications are normally presented in full-screen mode. The single window can be either full-screen or have window borders and controls available. You can change between these two modes by using ALT+ENTER. If the resolution selected matches the screen size, the application covers the entire screen; otherwise, it is a resizable window with scroll bars. You cannot open additional windows in a standard graphics application. Standard graphics applications have neither a menu bar at the top of the window, nor a status bar at the bottom. Fortran standard graphics applications are appropriate for problems that:
l l
Require numerical processing and some graphics Do not require a sophisticated user interface
When you select the Fortran standard graphics project type, the visual development environment includes the QuickWin library automatically, which lets you use the graphics functions. When building from the command line, you must specify the /libs:qwins option. You cannot use the run-time functions meant for multiple-window projects if you are building a standard graphics project. You cannot make a Standard Graphics application a DLL. For more information about Standard Graphics (QuickWin single window) applications, see Using Quickwin.
Page 10 of 58
Fortran QuickWin graphics applications (.EXE) have a multiple-document interface. Applications that use a multiple-document interface (MDI) have a menu bar at the top of the window and a status bar at the bottom. The QuickWin library provides a default set of menus and menu items that you can customize with the QuickWin APIs. An application that uses MDI creates many "child" windows within an outer application window. The user area in an MDI application is a child window that appears in the space between the menu bar and status bar of the application window. Your application can have more than one child window open at a time. Fortran QuickWin applications can also use the DFLOGM.F90 module to access functions to control dialog boxes. These functions allow you to display, initialize, and communicate with special dialog boxes in your application. They are a subset of Win32 API functions, which Windows applications can call directly. For more information on using dialog boxes, see Using Dialogs. When you select the Fortran QuickWin project type, the visual development environment includes the QuickWin library automatically, which lets you use the graphics functions. QuickWin applications should be written as multithreaded applications. (For information about multithreaded programs, see Creating Multithread Applications.) When building from the command line, you must specify the /libs:qwin compiler option to indicate a QuickWin application (or /libs:qwins option to indicate a Fortran Standard Graphics application). A Fortran QuickWin application that uses the compiler option is similar to a Fortran Standard Graphics application in that it has no menu bar or status bar. (A Fortran Standard Graphics application is a QuickWin application with a set of preset options that is offered in the program types list for your convenience.) As with a Fortran Standard Graphics application, the application covers the entire screen if the resolution selected matches the screen size; otherwise, it is a resizable window with scroll bars. You cannot make a Fortran QuickWin application a DLL. For information on how to use QuickWin functions, including how to open and control multiple windows, see Using Quickwin.
Page 11 of 58
Although you can call some of the Win32 APIs from the other project types, Fortran Windows applications allow you to use the full set of Win32 routines and use certain system features not available for the other project types. The DFWIN.F90 module contains interfaces to the most common Win32 APIs. If you include the USE DFWIN statement in your program, nearly all of the Win32 routines are available to you. The DFWIN.F90 module gives you access to a full range of routines including window management, graphic device interface, system services, multimedia, and remote procedure calls. Window management gives your application the means to create and manage a user interface. You can create windows to display output or prompt for input. Graphics Device Interface (GDI) functions provide ways for you to generate graphical output for displays, printers, and other devices. Win32 system functions allow you to manage and monitor resources such as memory, access to files, directories, and I/O devices. System service functions provide features that your application can use to handle special conditions such as errors, event logging, and exception handling. Using multimedia functions, your application can create documents and presentations that incorporate music, sound effects, and video clips as well as text and graphics. Multimedia functions provide services for audio, video, file I/O, media control, joystick, and timers. Remote Procedure Calls (RPC) gives you the means to carry out distributed computing, letting applications tap the resources of computers on a network. A distributed application runs as a process in one address space and makes procedure calls that execute in an address space on another computer. You can create distributed applications using RPC, each consisting of a client that presents information to the user and a server that stores, retrieves, and manipulates data as well as handling computing tasks. Shared databases and remote file servers are examples of distributed applications. Writing Fortran Windows applications is much more complex than other kinds of Visual Fortran projects. For more information on how to create Fortran Windows applications, see Creating Windows Applications. You can access the Windows API online documentation help file, Platform SDK, included with Visual Fortran. You can also obtain information through the Microsoft Developer Network. Microsoft offers Developer Network membership, which includes a development library and a quarterly CD containing technical information for Windows programming. For information on calling Win32 routines, see Calling Win32 Routines.
Page 12 of 58
Fortran static libraries (.LIB) are blocks of code compiled and kept separate from the main part of your program. The Fortran static library is one of the Fortran project types. Static libraries offer important advantages in organizing large programs and in sharing routines between several programs. These libraries contain only subprograms, not main programs. A static library file has a .LIB extension and contains object code. When you associate a static library with a program, any necessary routines are linked from the library into your executable program when it is built. Static libraries are usually kept in their own directories. If you use a static library, only those routines actually needed by the program are incorporated into the executable image (.EXE). This means that your executable image will be smaller than if you included all the routines in the library in your executable image. Also, you do not have to worry about exactly which routines you need to include the Linker takes care of that for you. Because applications built with a static library all contain the same version of the routines in the library, you can use static libraries to help keep applications current. When you revise the routines in a static library, you can easily update all the applications that use it by relinking the applications. A static library is a collection of source and object code defined in the FileView pane. The source code is compiled when you build the project. The object code is assembled into a .LIB file without going through a linking process. The name of the project is used as the name of the library file by default. If you have a library of substantial size, you should maintain it in a dedicated directory. Projects using the library access it during linking. When you link a project that uses the library, selected object code from the library is linked into that project's executable code to satisfy calls to external procedures. Unnecessary object files are not included. When compiling a static library from the command line, include the /c compiler option to suppress linking. Without this option, the compiler generates an error because the library does not contain a main program. To debug a static library, you must use a main program that calls the library routines. Both the main program and the static library should have been compiled using the debug option. After compiling and linking is completed, open the Debug menu and choose Go to reach breakpoints, use Step to Cursor to reach the cursor position, or use the step controls on the Debug toolbar.
Page 13 of 58
Using Static Libraries You add static libraries to a main project in the visual development environment with the Add to Project, Insert Files option in the Project menu. You can enter the path and library name in the Insert Files into Project dialog box with a .LIB extension on the name. If you are using a foreign makefile, you must add the library by editing the makefile for the main project. If you are building your project from the command line, add the library name with a .LIB extension and include the path specification if necessary. For an example of a static library project, see the Visual Fortran Samples folder ...\DF98\Samples\Scigraph\Scigraph, which creates a static library. To create a static library from the command line, use the /c compiler option to suppress linking and use the LIB command (see Managing Libraries with LIB). To create a static library from the visual development environment, specify the Fortran Static Library project type.
Page 14 of 58
DLL instead of in the main application file. In the visual development environment, after you open a workspace: From the Project menu, click Settings to display the project settings dialog box Click the Fortran tab Select the Library category In the Use Fortran Run-Time Libraries box, select DLL From the command line, use the /libs:dll compiler option to build applications with the run-time library stored in a separate DLL. For more information about DLLs, see:
l
To create a new project: 1. Click the File menu and select New. A dialog box opens that has the following tabs: Files Projects Workspaces Other Documents 2. The Projects tab displays various project types. Click the type of Fortran project to be created. If you have other Visual tools installed, make sure you select a Fortran project type. You can set the Create New Workspace check box to create a new Workspace. 3. Specify the project name and location. 4. Click OK to create the new project. Depending on the type of project being created, one or more dialog boxes may appear allowing you to only create the project without source files or create a template-like source file. If a saved Fortran environment exists for the Fortran project type being
Page 15 of 58
created, you can also import a Fortran environment to provide default project settings for the new project (see Saving and Using the Project Setting Environment for Different Projects). This action creates a project workspace and one project. It also leaves the project workspace open. To discontinue using this project workspace, click Close Workspace from the File menu. To open the project workspace later, in the File menu, click either Open Workspace or Recent Workspaces. To add files to an existing project:
l
To add an existing file to the project: 1. If not already open, open the project workspace (use the File menu). 2. In the Project menu, select Add to Project and click Select Files... from the submenu. 3. The Insert Files into Project dialog box appears. Use this dialog box to select the Fortran files to be added to the Project. To add more than one file to the project, hold down the Ctrl key as you select each file name. To add a new file to the project: 1. If not already open, open the project workspace (use the File menu). 2. In the Project menu, select Add to Project and click New... from the submenu. 3. The New dialog box appears. Specify the file name and its location. 4. Click (select) the type of file (Fortran Fixed Format Source or Fortran Free Format Source). 5. Click OK. The editor appears, letting you type in source code. The file name appears in the FileView pane.
To define a project from a set of existing or new source files: 1. 2. 3. 4. 5. On the File menu, click New... Click the Projects tab. Select the type of project. Name the project. Click OK. Depending on the type of project being created, one or more dialog boxes may appear allowing you to only create the project without source files or create template-like source files. If a saved Fortran environment exists for the project type being created, you can also import a Fortran environment to provide default project settings for the new Fortran project (see Saving and Using the Project
Building Programs and Libraries Setting Environment for Different Projects). 6. Add existing files and/or new files to the project. To add an existing file to the project:
Page 16 of 58
1. In the Project menu, select Add to Project and click Files... from the submenu. 2. The Insert Files into Project dialog box appears. Use this dialog box to select the Fortran files to be added to the Project. To add more than one file to the project, hold down the Ctrl key as you select each file name. To add each new file to the project: 1. In the Project menu, select Add to Project and click Select New... from the submenu. 2. The New dialog box appears. Specify the file name and its location. 3. Click the type of file (Fortran Fixed Format Source or Fortran Free Format Source). 4. Click OK. The editor appears allowing you to type in source code. The file name appears in the FileView pane. 7. You can now select "Build filename" from the Build Menu to build your application. You need to add these kinds of files to your project:
l l l
Program files with .F90, .FOR, or .F extension A resource file with an .RC extension If your project references routines or data in a Fortran dynamic-link library (DLL), you need to add the import library (.LIB file created while building the DLL) as a file to your project (see Building and Using Dynamic-Link Libraries)
How to use icon files, see Using QuickWin. Using binary files, see Files, Devices, and I/O Hardware. Using the Dialog Editor, Graphics Editor, or other Resource Editors, see the Visual C++ User's Guide.
Files in a Project Selecting a Configuration Setting Build Options Specifying Consistent Library Types
Page 17 of 58
Files in a Project
When you create a project, the Microsoft visual development environment always creates the following files: File Project workspace file Project file Workspace options file Extension .DSW .DSP Description Stores project workspace information. Used to build a single project or subproject. Contains environment settings for Visual Fortran, such as window sizes and positions, insertion point locations, state of project breakpoints, contents of the Watch window, and so on.
.OPT
Directly modifying the .DSW and .DSP files with a text editor is not supported. For information on creating (exporting) a makefile, see The Project Makefile. When you create a project, you also identify a project subdirectory. If the subdirectory does not exist, the visual development environment creates it. Project files that the visual development environment creates are put into this directory. When you create a project, the visual development environment also specifies subdirectories for intermediate and final output files for the various configurations that you specify. These subdirectories allow you to build configurations without overwriting intermediate and final output files with the same names. The General tab in the Project Settings dialog box lets you modify the subdirectories, if you choose. If you have existing source code, you should organize it into directories before building a project, although it is easy to move files and edit your project definitions if you should later decide to reorganize your files. If your program uses modules, you do not need to explicitly add them to your project, they appear as dependencies (.mod files). For a newly created project (before you request a Update Dependencies), the visual development environment scans the file list for modules and compiles them before program units that use them. The visual development environment
Page 18 of 58
automatically scans the added project files recursively for modules specified in USE statements, as well as any INCLUDE statements. It scans both source files (.FOR, .F, .F90) and the resource file (.RC), and adds all the files it finds to a Dependencies folder. You cannot directly add or delete the files listed in this folder. Module files are a precompiled, binary version of a source file, stored as a .mod file. When you change the source definition of a module, you can update the .mod file before you rebuild your project by compiling that source file separately by clicking (selecting) the file in the FileView pane and select Compile from the Build menu. To control the placement of module files in directories, use the /[no]module compiler option. To control the search for module files in directories, use the /[no]include and the /assume:source_include compiler options. The Project Makefile The visual development environment speeds and simplifies the task of building programs and libraries outside of the visual development environment by allowing you to export a makefile, which is a set of build instructions for each project. Makefiles contain the names of the source, object, and library files needed to build a program or library, plus the compiler and linker options selected in the Project Settings dialog boxes. The visual development environment updates the build instructions in internal makefiles when you add or remove project files in the project window, and when you make changes to the compiler or linker options in the Project Settings dialog boxes. To get an updated version of a makefile, update project dependencies (use the Build menu) and from the Project menu, select Export Makefile. The makefile is used by the external program maintenance utility, NMAKE.EXE. You can edit the makefile generated by the visual development environment if you need to perform unusual or exceptional builds. Remember, however, that once you have edited a makefile, exporting the makefile again from the visual development environment will overwrite your changes. If you use a foreign makefile for a project, the visual development environment calls NMAKE to perform the build. You can run NMAKE from the console command line to perform builds either with makefiles exported by the visual development environment or with foreign makefiles that you have edited. For more about the external program maintenance utility, see Building Projects with NMAKE.
Page 19 of 58
Note: When you use a foreign makefile, the project is considered to be foreign. You cannot use the Project Settings dialog box to make changes to the build options, or use the Add to Project dialog box to add files.
Selecting a Configuration
A configuration defines the final binary output file that you create within a project. When you create a new project, the visual development environment creates the following configurations: Debug configuration By default, the debug configuration sets project options to include the debugging information in the debug configuration. It also turns off optimizations. Before you can debug an application, you must build a debug configuration for the project. The release configuration does not include the debugging information, and it uses any optimizations that you have chosen.
Release configuration
Select the configuration in the Build menu, Set Active Configuration item. Only one configuration can be active at one time. When you build your project, the currently selected configuration is built:
l
If you selected the debug configuration, a subfolder called Debug contains the output files created by the build for the debug version of your project. If you selected the release configuration, a subfolder called Release contains the output files created by the build for the release version of your project.
Project type: specifies the type of Fortran application to build, such as a Fortran Static Library, Fortran Console application, Fortran QuickWin application, Fortran Windows application, and so on. Build options: specifies the build options, which include the compiler and linker options (project settings).
Although debug and release configurations usually use the same set of source files, the information about project settings usually differs. For example, the default debug configuration supplies full debug information and no optimizations, whereas the default release configuration supplies minimal debug information and full optimizations. You can also define new configurations within your project. These configurations
Page 20 of 58
can use the existing source files in your project, the existing project settings, or other characteristics of existing configurations. A new configuration does not have to share any of the characteristics or content of existing configurations, however. You could, for instance, create an initial project with debug and release configurations specifying an application for the Win32 environment, and add source files to the project. Later, within the project, you could create debug and release configurations specifying a DLL for the Win32 environment, add an entirely disjoint set of files to this configuration, and make these configurations dependencies of the application configurations. Platform Types The platform type specifies the operating environment for a project. The platform type sets options required specifically for a given platform, such as options that the compiler uses for the source files, the static libraries that the linker uses for the platform, the default locations for output files, defined constants, and so on. Visual Fortran supports the Win32 platform type. For more information:
l
On viewing and changing the project build options for the current configuration, use the Project settings dialog box (see Setting Build Options). On errors during the build process, see Errors During the Build Process.
Page 21 of 58
The FileView pane shows the files associated with the project configuration and allows you to select certain files. You can set some types of options, such as linking or requesting browse information, only at the configuration level. You can set options at the following levels within a configuration: Any options set for the current configuration apply to every file in the configuration unless overridden at the file level. Options set for the configuration apply to all actions, such as compilation, linking, and requesting browser information. Any options set for a file apply only to that file and override any options set at the configuration level. Options set for selected files apply to file-level actions, such as compiling.
Configuration level
File level
You can insert both source files (.FOR, .F90, .F, .FI, .FD) and object files (.OBJ) by using the Project menu Add to Project, Files item. You should always insert all source files used by your application into the project. For example, when you update a source file, the next time you build your application, the visual development environment will create a new object file and link it into your project. You should also insert the names of any necessary static libraries and DLLs with .LIB extensions to be linked with your project. Use only the library names, not the names of any files within the libraries. If you have installed the same version of Microsoft Visual C++ and Visual Fortran (in the same directory tree), you can include C/C++ source code files. If the same version of Microsoft Visual C++ is not installed, include C/C++ object code instead. You can set and save project settings as described in the following sections:
l l l
Compile and Link Options for a Configuration or Certain Files Saving and Using the Project Setting Environment for Different Projects Source Browser Information for a Configuration
Page 22 of 58
1. Open your project workspace. 2. In the Project menu, click Settings. The Fortran tab of this dialog box presents options grouped under different categories. Select the category from the Category drop-down list:
You can choose compiler and linker options through the various categories in the Fortran and Link tabs. For details about the compiler options, see Categories of Compiler Options. If a compiler option is not available in the Project Settings dialog box you can enter the option in the lower part of the window under Project Options: just as you would using the command line. The FileView pane shows the files associated with the project configuration and lets you select certain files. The options that you set at the file level in the configuration override options set at the configuration level. Linking options only be applied at the configuration level (not the file level). The linker builds an executable program (.EXE), static library (.LIB), or dynamic-link library (.DLL) file from Common Object File Format (COFF) object files and other libraries identified in the linker options. You direct the linker by setting linker options either in the visual development environment, in a build instructions file, or on the console command line. For example, you can use a linker option to specify what kind of debug information to include in the program
Page 23 of 58
For more information on compiler and linker options, see Compiler and Linker Options.
Saving and Using the Project Setting Environment for Different Projects
You can set any of the compiler or linker options described in Compiler and Linker Options in the Project menu, Settings dialog box. The Fortran tab of this dialog box presents several categories of options to set. Visual Fortran provides a facility to save and re-use the Project settings for multiple projects of the same project type. For example, you can save your Project settings environment for a Fortran QuickWin project and use those saved settings as defaults when you create a new Fortran QuickWin project type. The saved Fortran environment project settings are those associated with the Fortran tab and the displayed tool tab(s) in the Project Settings dialog box. You can:
l l l
Save a Fortran project settings environment for an existing project Use a previously saved project settings environment for a new project Manage the current set of saved Fortran environment settings
To save a project settings Fortran environment: 1. Open the appropriate workspace. 2. Modify the Project Settings dialog box as needed. If you specify actual file names for output files, consider changing them to use the default file naming conventions. 3. In the File menu, click Save Fortran Environment or click the green tree on the Fortran toolbar. A window resembling the following dialog box appears: Saving a Project Settings Environment
Page 24 of 58
4. The Tool Combo box allows you to view the project settings for either the Fortran or the displayed tools (such as Linker). The Release and Debug configuration values are displayed for the selected tab. Verify that the displayed values are acceptable. 5. The edit box titled Saved Fortran Console Environment allows you to specify the name of the environment to be saved. 6. Click the Save Settings button to save the settings as a project settings environment. To use an existing Fortran environment when creating a new project: 1. If a Fortran environment exists for the specified new Fortran project type, you will be asked whether you want to apply project settings options from a saved Fortran environment. If you click Yes, a window resembling the following dialog box appears: Using an Existing Fortran Project Settings Environment
Page 25 of 58
2. For the selected Fortran project type, a list of saved Fortran environments appears. Select a Fortran environment. Verify that the selected environment is correct by viewing the Project Settings options. 3. After selecting the appropriate Fortran environment for the Fortran project being created, click the Apply button to use the saved settings for the new project. 4. Complete other tasks associated with creating a new project, such as adding source files, and so on. To manage saved Fortran environments: 1. In the Tools menu, click Managed Saved Fortran Environment or click the saw on the Fortran toolbar. A dialog box resembling the following appears: Managing Saved Fortran Project Settings Environments
Page 26 of 58
2. Initially, this dialog box displays the project types for which there are saved Fortran environments. Double-click on the project type name to view the saved Fortran environments for that project type. This dialog box allows you to display the Fortran environments associated with each project type. Double-click the name of a project type to display the Fortran environments associated with that project type. 3. To display the project settings for a Fortran environment: Click the name of a Fortran environment. View the project settings for the Fortran tab. Click the other tool tab (such as Linker) and view that tool's project settings. If needed, click (select) a different Fortran environment. 4. To determine whether a duplicates exist for a Fortran environment: Click (select) the name of an environment or a project type. Click the Display Environments with Duplicate Settings button. If the Fortran environments have different project settings, No Duplicates Found is displayed. If the Fortran environments have identical project settings, the duplicate environments are displayed. 5. To delete a Fortran environment: Click (select) the name of an environment or the project type. Click the Delete button. Click OK to the delete confirmation box.
In the visual development environment: 1. In the Project menu, click Settings. 2. In the General category of the Fortran tab, set the Generate Source Browse Information check box. 3. Click the BrowseInfo tab and set the Build Browse info check box.
Page 27 of 58
Click OK. Build your application. In the Tools menu, click Source Browser. Near the bottom on the Browse window, locate the Case sensitive check box. Since Fortran is a case-insensitive language, make sure the Case sensitive check box is clicked off. 8. When you are done using the Browse window, click OK. The Browse window allows you to view graphs of calling relationships between functions and view the symbols contained within the file, and perform other functions.
l
On the command line: 1. Specify the /browser option. 2. Use the Browse Information File Maintenance Utility (BSCMAKE) utility to generate a browse information file (.BSC) that can be examined in browse windows in the visual development environment.
When the browse option is on, the compiler creates intermediate .SBR files when it creates the .OBJ files; at link time, all .SBR files in a project are combined into one .BSC file. These files are binary, not readable, but they are used when you access them through the Browser menu.
Mixing single-threaded with multi-threaded versions of the libraries Mixing static and dynamic-link versions of the libraries Mixing debug with non-debug versions of the libraries
The default Fortran libraries depend on the project type: Fortran Project Type Fortran Console Fortran Standard Graphics Fortran QuickWin Default Libraries Used Static, single-threaded libraries dfor.lib and libc.lib Static, multithreaded libraries dformt.lib and libcmt.lib Static, multithreaded libraries dformt.lib and libcmt.lib
Building Programs and Libraries Fortran Windows Fortran DLL Fortran Static Library
Page 28 of 58
Static, multithreaded libraries dformt.lib and libcmt.lib Dynamic-link libraries dfordll and msvcrt (and their import libraries) Static, single-threaded libraries dfor.lib and libc.lib
Pure Fortran applications can have mismatched types of libraries. One common scenario is a Fortran QuickWin application that links with a Fortran Static library. Fortran QuickWin (and Fortran Standard Graphics) applications must use the static, multithreaded libraries, and by default, Fortran Static libraries are built using static, single-threaded libraries. This causes a conflict, so the Fortran Static library and the QuickWin application must both be built using static, multithreaded libraries. Similarly, different C/C++ applications link against different C libraries. If you mix the different types of applications without modifying the defaults, you can get conflicts. The debug version of a library has a letter "d" appended to its base file name:
l l l
Static, single-threaded: libc.lib and libcd.lib Static, multithreaded: libcmt.lib and libcmtd.lib Dynamic-link libraries: msvcrt and msvcrtd (.lib import library and .dll)
When using a Debug configuration, Visual C++ selects the debug libraries. Visual Fortran does not select debug libraries for any configuration, but provides a Project Settings option to request their use. To specify different types of Fortran libraries in the visual development environment, open the Project Settings dialog box, click the Fortran tab, and select the Libraries category:
l
To specify static libraries, select the appropriate type of static (non-DLL) library under Use Run-Time Libraries (see /libs:static). To specify dynamic-link libraries, select the appropriate type of DLL library under Use Run-Time Libraries (see /libs:dll). To specify multithreaded libraries, select the appropriate type of Multithread library under Use Fortran Run-Time Libraries (see /threads). To specify the debug libraries, select the appropriate type of Debug library under Use Run-Time Libraries (see /[no]dbglibs). If you specify debug libraries (/dbglibs) and also request DLL libraries (/libs:dll) (Use Run-Time Libraries that include both Debug and DLL), be aware that this combination selects the debug versions of the Fortran DLLs. These Fortran DLL files have been linked against the C debug DLLs. When you specify QuickWin or Standard Graphics libraries under Use RunTime Libraries, this selection implicitly requests Multi-threaded libraries.
For details about the Visual C++ libraries used and compiler options available,
Building Programs and Libraries see Visual Fortran/Visual C++ Mixed-Language Programs.
Page 29 of 58
Compile a file without linking Build a project Rebuild all parts of a project Batch build several configurations of a project Clean extra files created by project builds Execute the program, either in debug mode or not Update program dependencies Select the active project and configuration Edit the project configuration Define and perform profiling
When you have completed your project definition, you can build the executable program. When you select Build projectname from the Build menu (or one of the Build toolbars), the visual development environment automatically updates dependencies, compiles and links all files in your project. When you build a project, the visual development environment processes only the files in the project that have changed since the last build. The Rebuild All mode forces a new compilation of all source files listed for the project. You either can choose to build a single project, the current project, or you can choose multiple projects (requires batch build) to build in one operation. You can execute your program from the visual development environment using Ctrl+F5 or Execute from the Build menu (or Build toolbar), or from the command line prompt. Compiling Files In a Project You can select and compile individual files in any project in your project workspace. To do this, select the file in the project workspace window (FileView tab). Then, do one of the following:
l
Press Ctrl+F7. or
Page 30 of 58
Click the right mouse button to display the pop-up menu and select Compile.
You can also use the Ctrl+F7 or Compile from the Build menu (or Build toolbar) options when the source window is active (input focus).
Page 31 of 58
To view the include file and library directory paths in the visual development environment: 1. In the Tools menu, click Options. 2. Click the Directories tab. 3. In the drop-down list for Show Directories For, select Include files and view the include file paths. 4. In the drop-down list for Show Directories For, select Library files and view the library paths. 5. Click OK if you have changed any information. To view the libraries being passed to the linker in the visual development environment: 1. If not already open, open your Project Workspace (File menu, Open Workspace). 2. In the Project menu, click Settings. 3. Click the Link tab 4. In Link tab's Project Options box, type the following option:
/verbose:lib
5. Click OK. The next time you build the project, the list of libraries will be displayed with other build-related messages in the output pane (bottom of screen). If you have trouble linking CXML libraries, see also Using CXML from Visual Fortran. With the Professional and Enterprise Editions, if you have trouble linking IMSL libraries, specify the /imsl option and see Using the IMSL Libraries from Visual Fortran. If you encounter linker errors, see Linker Diagnostic Messages and Error Conditions. To view a description of build and run-time messages, see Error Messages or, in the visual development environment, highlight the error identifier (such as LNK2001) in the output pane and press F1.
Compiler Limits
The following table lists the limits to the size and complexity of a single Visual Fortran program unit and to individual statements contained in it. The amount of data storage, the size of arrays, and the total size of executable
Page 32 of 58
programs are limited only by the amount of process virtual address space available, as determined by system parameters: Language Element Actual number of arguments per CALL or function reference Arguments in a function reference in a specification expression Array dimensions Array elements per dimension Constants; character and Hollerith Constants; characters read in listdirected I/O Continuation lines DO and block IF statement nesting (combined) DO loop index variable Format group nesting Fortran source line length INCLUDE file nesting Labels in computed or assigned GOTO list Lexical tokens per statement Named common blocks Parentheses nesting in expressions Structure nesting Symbolic name length Limit No limit No limit 7 9,223,372,036,854,775,807 or process limit 2000 characters 2048 characters 511 128 9,223,372,036,854,775,807 or process limit 8 132 characters 20 levels 500 3000 250 40 20 63 characters
Page 33 of 58
Page 34 of 58
("filename"), but this is not required unless the argument contains spaces or tabs. A null argument consists of an empty set of double quotation marks with no filename enclosed (""). The following example runs the program MYPROG.EXE from the console:
MYPROG "" OUTPUT.DAT
Because the first filename argument is null, the first OPEN statement with a blank filename field produces the following message:
File name missing or blank - please enter file name<R> UNIT number ?
The number is the unit number specified in the OPEN statement. The filename OUTPUT.DAT is used for the second such OPEN statement executed. If additional OPEN statements with blank filename fields are executed, you will be prompted for more filenames. Programs built with the QuickWin library prompt for a file to open by presenting a dialog box in which you can browse for the file or type in the name of a new file to be created. Instead of using the /fpscomp:filesfromcmd option, you can:
l
Call the GETARG library routine to return the specified command-line argument. To execute the program in the visual development environment, provide the command-line arguments to be passed to the program in the Project menu Settings item, Debug tab, in the Program Arguments text box. Call the GetOpenFileName Win32 routine to request the file name using a dialog box. For an example of how to call the GetOpenFileName routine, see the Visual Fortran Sample GetOpenFileName in ...Df98 \Samples\Advanced\Win32\GetOpenFileName.
If you use the visual development environment debugger and need to specify a working directory that differs from the directory where the executable program resides, specify the directory in the Project menu Settings item, Debug tab, in the Working Directory text box. Run-time error messages are displayed in the console or in a message box depending upon the project type of the application you build (see Run-Time Message Display and Format). If you need to capture these messages, you can redirect stderr to a file. For example, to redirect run-time error messages from a program called BUGGY.EXE to a file called BUGLIST.TXT, you would use the following syntax:
BUGGY.EXE > BUGLIST.TXT
The redirection portion of the syntax must appear last on the command line.
Page 35 of 58
You can append the output to an existing file by using two greater-than signs (>>) instead of one. If the file does not exist, one is created. For more information about command-line redirection, see Redirecting Command-Line Output to Files. To run (deploy) your application on another Windows system, use the VFRUN tool to install the necessary run-time DLLs and related files. The VFRUN tool provides the run-time redistributable files for Visual Fortran applications, in the form of a self-installing kit. You can download the run-time redistributable files VFRUN kit from the Visual Fortran Web page (click Downloads area). VFRUN and the deployment of Visual Fortran applications is further described in the Compaq Visual Fortran Installing and Getting Started guide. For more information:
l
On using the debugger with Fortran programs, see Debugging Fortran Programs. Locating errors in the debugger, see Locating Run-Time Errors in the Debugger. On locating the source of exceptions, see Locating Run-Time Errors and Using Traceback Information. On handling run-time errors with source changes, see Methods of Handling Errors On environment variables recognized during run-time, see Run-Time Environment Variables On each Visual Fortran run-time message, see Run-Time Errors
Copying Projects
You need to follow certain procedures to move a project's location, whether you copy a project to:
l l
Another disk or directory location on the same PC system Another PC system where Visual Fortran is also installed
To copy an existing Visual Fortran project to another disk or system: 1. Copy all project files to the new location. You do not need to copy the subdirectories created for each configuration. Keep the folder/directory hierarchy intact by copying the entire project tree to the new computer. For example, if a project resides in the folder \MyProjects\Projapp on one computer, you can copy the contents of that directory, and all subdirectories, to the \MyProjects\Projapp directory on another computer. 2. After copying all of the files, delete the following files from the main
Page 36 of 58
directory at the new location. These files are disk- and computer-specific and should not be retained: *.opt files *.ncb files (if present) 3. After copying the files, if you copied the subdirectories associated with each configuration, you need to delete these contents of subdirectories at the new location, such as the Debug and Release subdirectories. The files contained in these subdirectories are disk- and computer-specific files and should not be retained. For example, Visual Fortran module (.MOD) files contained in these subdirectories should be recreated by the Visual Fortran compiler, especially if a newer version of Visual Fortran has been installed. The internal structure of module files can change for Visual Fortran releases. If you copied the project files to the same system or a system running the same platform or major Visual Fortran version, do the following steps to remove most or all of the files in the configuration subdirectory: Open the appropriate project workspace. In the File menu, either select Open Workspace or select Recent Workspaces. If you use Open Workspace, select the appropriate .DSW file. For each configuration, select the appropriate configuration by selecting Set Active Configuration in the Build menu. For the current configuration, select Clean in the Build menu. Repeat the previous two steps for other configurations whose subdirectories have been copied. 4. If you copied the project from a PC running a Visual Fortran 5, you should open the project at its new location using Visual Fortran 5, before you convert the project to Visual Fortran Version 6 format. If you open a project created with Visual Fortran 5 with Visual Fortran Version 6, Visual Fortran 6 will convert the project to Visual Fortran Version 6 format. Converting the project back to Version 5 format is not easily performed (must be done manually). If possible, open the project under Visual Fortran Version 5 after you move it, before you convert the project to Version 6 format. This ensures that the project has been moved successfully and minimizes the chance of conversion problems. An alternative is to make an archive copy of the Visual Fortran 5 project files before you open and convert the project to Visual Fortran Version 6 format. 5. View the existing configurations. To view the existing configurations associated with the project: Open the project workspace. In the File menu, either select Open Workspace or select Recent Workspaces. If you use Open Workspace, specify the appropriate .DSW file (navigate to the appropriate project directory and double-click the .DSW file). Select Configurations from the Build menu.
Page 37 of 58
View the available configurations. You can also remove or add configurations. 6. If you copied the project to a PC running a different platform, specify new configurations. If you copied the project to the same system or to a system where the same major version of Visual Fortran is installed that uses the same platform, you may not need to create any new configurations. If you copy the files to a different platform, opening the project reveals that the target platform is still set to the original platform. Although this is not obvious, you can tell this is so because the Build, Compile, and Execute options are grayed out in the Build menu. Before you can build the application on the new platform, you must first specify one or more new configurations for the project on the new platform. To create Debug and Release targets for this project, you create a new configuration while running Visual Fortran on the new platform. The platform for a new configuration is assumed to be the current platform. This same behavior applies when moving projects between any two platforms. To create a new project configuration: a. In the Configurations dialog box, click the Add button. The Add Project Configuration dialog box appears. b. In the Configuration box, type a new configuration name. The names do not matter, as long as they differ from existing configuration names. c. Select the configuration from which to copy the settings for this configuration and click OK. Usually, you will want to copy the settings from a similar configuration. For example, if this new configuration is a release configuration, you will usually copy settings from an existing release configuration. d. The Projects dialog box appears with the new project configuration. Repeat the process as necessary to create as many configurations as you need. 7. Check and reset project options. Because not all settings are transportable across different disks, systems, or platforms, you should verify your project settings on the new platform. To verify your project settings: a. From the Project menu, choose Settings. The Project Settings dialog box appears. b. Review the tabs and categories to ensure that the project settings you want are selected. Pay special attention to the following items:
Page 38 of 58
General Tab Review the directories for intermediate and output files. If you moved the project to a different system, be aware that any absolute directory paths (such as C:\TEMP or \Myproj\TEMP) will most likely need to be changed. Instead, use relative path directory names (without a leading back slash), such as Debug. Custom Build Tab Review for any custom commands that might change between platforms. Fortran and Linker tabs Nonstandard options in the original configurations must be replicated (as applicable) in the new configurations.
For example, if you moved a project from a Windows NT Alpha system running Visual Fortran 6.5A or earlier, a few Alpha-only compiler options are either not needed (such as /synchronous_exceptions and /granularity) or are not available on ia32 systems (such as /rounding_mode, /pipeline, and /math_library:accurate). You also might find new compiler options that might benefit your application (see Compiler Options). n Pre-link and Post-build Step tabs Review for any custom commands that might change between platforms. 8. If you are copying your project between systems, check your source code for directory paths referenced in INCLUDE or similar statements. Developer Studio provides a multi-file search capability called Find in Files, available from the Edit menu.
Building Programs and Libraries (after installation): To Locate: Roadmap to the Samples Look:
Page 39 of 58
Samples folders
For example, after a Custom installation, the ...\Df98\Samples\Tutorial folder contains short example programs, called Tutorial samples. The Tutorial samples describe (as source comments) how they can be built. Unlike other samples, all Tutorial samples are contained in a single directory. Longer sample programs are also provided in their own subdirectories and include a makefile (for command-line use) as well as the source files. Many samples include a project workspace file, allowing you to open the project workspace in the visual development environment, view the source files in the FileView pane, build the sample, and run it. Samples for the Compaq Array Visualizer (Professional and Enterprise Editions) are located in folders under ...\ArrayVisualizer\Samples. For a description of the Samples, see the Roadmap to the Visual Fortran Samples.
Page 40 of 58
l l l l l l
l l l
Advanced Samples: Samples Using Component Object Model Samples Showing Direct Fortran 90 Descriptor Manipulation Samples Using OpenGL Samples Using Win32 API Calls Dialog Box Samples DLL Samples Exception Handling ISO Varying Strings Mixed-Language Programming Samples QuickWin Programming Samples Includes the Postmini graphical postprocessing software, which can be used to examine the output of a number of process, device and circuit simulation (TCAD) programs, as well as import data from ASCII files. Scientific Graphing (SciGraph) Utility Samples Tutorial Samples (illustrate Fortran language constructs) Miscellaneous Samples: Includes a sample that converts data files created by Lahey Fortran to unformatted files which can be read by CVF
Advanced
Within the Advanced category, there are several other categories of samples. Each category is described below, and the specific samples within each are described in its section. Each category is contained within its own subdirectory of ...\Df98\Samples\Advanced:
l l
l l
Samples using Component Object Model (advanced\com) Samples Showing Direct Fortran 90 Descriptor Manipulation (advanced\descript) Samples Using OpenGL (advanced\opengl) Samples Using Win32 API Calls (advanced\win32)
Component Object Model (Advanced\Com) The COM samples demonstrate use of the Component Object Model from within Visual Fortran:
l
adder: The Adder sample demonstrates a Fortran COM server and its use by Visual Fortran, Visual C++, and Visual Basic clients. The COM server implements a very simple "adding machine" object. Each client displays a dialog box that uses the server. Adder is used as an example in Creating a COM Server. autodice: The autodice sample demonstrates interactions between Fortran
Page 41 of 58
and Excel 97. This program takes 12 numbers and uses Automation to tell Excel 97 to draw a histogram of the numbers. The EXCEL97A module generated by the Fortran Module Wizard is used as the interface to EXCEL. You must have Microsoft Excel 97 installed on your computer in order to successfully run this sample. dlines: The dlines sample demonstrates interactions between Fortran and the Developer Studio. This program converts Debug lines to IFDEF directives (metacommands). It uses COM to drive Microsoft Developer Studio to edit the source file. The COM interfaces defined in DSAPP.F90 are taken from:
...\Microsoft Visual Studio\Common\Msdev98\Bin\devshl.dll
The dlines sample program is invoked with the following command line:
dlines inputsource outputsource debugchar ifdefname
Where: Inputsource is the filename of the input source file. Note: The source file must be fixed form. outputsource is the filename of the output source file. debugchar is the the debug character in the source file to replace with IFDEF directives (metacommands) ifdefname is the name to use with the IFDEF directives This program is useful as a tool to edit PowerStation Fortran files that used something other than "C" or "D" as the comment character.
l
dsbuild: The dsbuild sample demonstrates using Automation to drive Developers Studio to rebuild a project configuration. The Automation interfaces defined in DSAPPA.F90 are taken from:
...\Microsoft Visual Studio\Common\Msdev98\Bin\devshl.dll
Where: project.dsw is the file specification of the project file. configuration is the name of the configuration to rebuild.
l
ErrorInfo: The ErrorInfo sample demonstrates a Fortran COM Server that supports COM ErrorInfo objects. For information about COM ErrorInfo objects, see Adding Support for COM ErrorInfo Objects and the Platform SDK online documentation. iweb: The iweb sample uses the COM interfaces in SHDOCVW.DLL to start a Web Browser and direct the browser to navigate to a specified URL. The iweb program is invoked with the following command line:
iweb url
Page 42 of 58
Any arguments not provided on the command line will be prompted for.
l
safearray: The safearray sample demonstrates converting a Fortran twodimensional array to a COM SafeArray and reading the data back from the SafeArray. variant: The variant sample demonstrates using the VARIANT manipulation routines in Visual Fortran. VARIANTs are used in COM interfaces.
Direct Fortran 90 Descriptor Manipulation (Advanced\Descript) In other languages, such as C or Pascal, pointers are simple memory addresses with no notion of how the information is organized at the address pointed to. Fortran 90 pointers differ dramatically, by not just containing a memory address, but also keeping track of the bounds and strides of each dimension of an array. To accomplish this, the Fortran 90 compiler uses something called an Array Descriptor, which stores the details of how the array is organized. Array Descriptors may also be called Dope Vectors in other implementations. By manipulating the contents of a descriptor directly, you can perform operations that would otherwise not be possible in strict Fortran 90. You can associate a Fortran 90 pointer with any piece of memory, organized in any way desired (so long as it is rectangular in terms of array bounds). You can also pass Fortran 90 pointers to other languages, such as C, and have the other language correctly interpolate the descriptor to obtain the information it needs. Important for mixed-language programming, you can create a Fortran 90 pointer associated with an array which is accessed in C's row order, rather than Fortran's traditional column order, allowing the same array to be used with the same subscripts from both C and Fortran. The descript workspace has three subprojects. Each creates its own executable, and there are no dependencies between them:
l
cassign: A Fortran 90 pointer is associated with a C array, in row order. This pointer is then passed back to Fortran, which reads the values written by C in exactly the same way that C wrote them. This project uses descript.h, which provides the C prototypes, routines, and structure definitions to the Fortran 90 pointers. fassign: A Fortran 90 pointer is associated with a Fortran 90 array, in row order, and then passed to a C routine which can read the array in natural C notation. cube: Through use of the Bitmap module a Fortran 90 pointer is created that points to an internal .BMP file, with its special structure (rows reversed). Once this pointer is setup, the .BMP can be written to directly and then quickly displayed on a QuickWin child window. This method is
Building Programs and Libraries used to show a flicker-free spinning cube. OpenGL (Advanced\Opengl)
Page 43 of 58
OpenGL is a protocol for performing advanced 3-dimensional graphics. Visual Fortran includes Fortran 90 modules and link libraries for using OpenGL from Fortran. The sample programs included in this section illustrate programs that take advantage of these facilities. All OpenGL samples will run under Windows NT, Windows 2000, Windows Me, and Windows 98, but require the OSR2 release of Windows 95 to run there. Refer to our Frequently Asked Questions at http://www.compaq.com/fortran/visual/faq.html for more information.
l
3dcube: The 3dcube sample displays a three-dimensional box on a black background. It uses QuickWin to create the window and draw the shapes using primitive 3D graphics. olympic: The olympic sample displays 5 rings on a window that rotate and move towards each other, ultimately creating the Olympic symbol. It illustrates calls to the OpenGL libraries, techniques for supporting code on 32- and 64-bit systems, techniques for reading arguments from the command line, and interface blocks, among other Fortran 90 concepts. This sample must be compiled with /fpscomp:logicals due to a difference in the way that Visual Fortran and Microsoft PowerStation Fortran interpret logical values. It must also be linked against the QuickWin libraries. puzzle: The puzzle sample displays a Rubik's Cube, and allows you to try and solve it. There is currently a problem with the colors in this sample. It uses Win32 calls to manage the windowing interface. Again, it illustrates techniques for supporting code on 32- and 64-bit systems. simple: The simple sample displays some points, lines, and simple shapes in a window. It uses QuickWin to manage the windowing interface. It is, as it sounds, a very simple example, using primitive 2-D graphics. One interesting aspect of it is the use of color and how to set some of the standard colors. wave: The wave sample displays a window with a surface waving. It uses QuickWin to manage the windowing interface. Beyond the usage of OpenGL, here are many interesting demonstrations in this sample, including the animation and the trapping of characters typed while the window has focus. The keys that are trapped are: C, which changes the contour of the wave [space], which pauses the animation. To resume, hit space again S, which changes the shading of the wave A, which causes the wave to spin. To stop spinning, hit A again L, which changes the color of the wave.
Page 44 of 58
Visual Fortran includes Fortran 90 modules and libraries for doing Windows programming at the Win32 API level. For information on calling Win32 routines from a Fortran application, see Calling Win32 Routines. These samples presented here show Win32 API examples of how this is done:
l
angle: The angle sample uses the Windows interface to manage the GUI. It illustrates a basic Windows program, including the event dispatcher, and the use of pens and dialog boxes. The angle sample displays a shape consisting of a straight line plus an arc, and allows the user to input X,Y coordinates, angle of the arc, and radius of the arc, and then redraws. bounce: The bounce sample uses Win32 calls but creates a console application, rather than a Windows based one. It is a multi-threaded application that creates a new thread whenever the letter "A" is typed, up to a maximum of 32. Each thread bounces a happy face of a different color across the screen. All threads are terminated when the letter "Q" is typed. This particular application requires that several compatibility switches be set: /fpscomp:logicals, /fpscomp:ioformat, /fpscomp:general. These are found under the Compatibility entry on the Fortran tab of the Project Settings dialog box. check_sd: The check_sd sample illustrates the use of security descriptors. The purpose of the sample is to assist people who want to manipulate security descriptors with sample code that they can start from. This sample, as is, examines the security descriptor on files, and could be easily modified to check the security descriptor on other objects. To use this sample, check_sd with no arguments checks the security descriptors on the A: device. check_sd d:\a.fil will check the d:\a.fil file. In this case, D: must be formatted with the NTFS file system because only NTFS files have security descriptors. cliptext: The cliptext sample demonstrates copying text to and from the clipboard. It uses the Windows interface to manage the GUI, and as such, also demonstrates the use of WinMain and MainWndProc that contains the event dispatcher. It has an "about" box, located under the File menu item, as there is no Help. cmndlg: The cmndlg sample demonstrates the use of common dialog boxes, including "About", File access dialogs, and error message dialogs. Each dialog box is demonstrated being used in three different ways: standard, using a modified template, and using a hook function. Cmndlg displays "Hello World" in a window, and allows the user to change the text and colors, etc. consolec: The consolec sample demonstrates the use of the SetConsoleTxtAttribute and GetConsoleScreenBufferInfo APIs to set or get the console text color attributes. This is a console application that is linked against dfwin.lib. If no arguments are passed to consolec, it prints out a color sample in a command window. The arguments are FOREGROUND color and BACKGROUND color, and the valid options are BLACK, BLUE, GREEN, CYAN, RED, MAGENTA, YELLOW or WHITE. The color argument
Page 45 of 58
passed must be in uppercase, or an error will be output. crypto: The crypto sample demonstrates using the Cryptography API to encrypt and decrypt files. cursor: The cursor sample demonstrates how to manipulate a cursor and select a region. It uses the Windows interface to manage the GUI. The program will either let the user draw boxes (regions) or will calculate 1027 prime numbers when RETURN is entered. During that calculation, the cursor is turned into an hourglass. drives: The drives sample determines all drives on the system, both local and remote, and determines their file system type. This is a console application that links against dfwin.lib. event: The event sample demonstrates the basic concepts involved when using asynchronous I/O. It reads one file asynchronously and writes another synchronously. This is a console application that links against dfwin.lib and requires two arguments, the input file and output file. This executable transfers one file to another. Refer to the readme.txt for information on how to set it up in the Developer Studio. exitwin: The exitwin sample demonstrates how to shut down applications. When run, it will ask whether you want to continue, as it will log you off. Optionally, you can compile it to reboot the system. This is a Windows application that uses Windows to manage the GUI. floppy: The floppy sample demonstrates how to access a physical floppy disk using the Win32 API set. It has two major functions: 1. It can be used to display the geometry of a disk, floppy -g a: 2. It can be used to produce a disk image or to write a disk image to a floppy, floppy -c a: bootdisk (produces a disk image of a:) or floppy -c bootdisk a: (make a: identical to bootdisk image). This is a Fortran Console program that is linked against dfwin.lib.
generic: The generic sample demonstrates the basic concepts required in building a Windows application and could be used as a basis for building a first Windows program. The specific features that generic implements are: 1. Custom icon 2. Standard Menu Bar 3. Standard Help Menu 4. Full WinHelp Support 5. Keyboard Accelerator Usage 6. Version Control Information getdev: The getdev sample demonstrates the GetDeviceCaps() API. Also, in this sample, the main window is a dialog box, and all the interesting code is in the window procedure for that dialog box. This is different than many of the other Windows-based samples, which use MainWndProc to handle the events. getopenfilename: The getopenfilename sample demonstrates the use of the API GetOpenFileName(). When run, it brings up a dialog box listing either text files or Fortran files in the project directory. It also gives you the
Page 46 of 58
option of opening the file read-only. getsys: The getsys sample demonstrates the APIs GetSysColor(), GetSystemDirectory(), GetSystemInfo(), GetSystemMetrics(), GetSystemPaletteEntries(), and GetSystemTime(). Also, in this sample, the main window is a dialog box, and all the interesting code is in the window procedure for that dialog box. This is different than many of the other Windows-based samples, which use MainWndProc to handle the events. inherit: The inherit sample demonstrates inheritance of file handles from one process to a child process. It also demonstrates the use of anonymous pipes as stdout and stderr replacements. It contains two projects, child and inherit. The inherit project creates inherit.exe, which can be run without the other project. To run inherit, specify two parameters: 1. Trace file to write stdout and stderr 2. Command to execute The inherit program is run within QuickWin so that it can be a "consoleless" parent. Otherwise, there are configuration problems when trying to pipe. To use this sample, first set the current project to child and build it, then set the current project to inherit and build it. Before you run it, read the readme.txt file in the project; this will tell you what changes you need to make to your project configuration in order to run this sample.
input: The input sample is a basic Windows application that does very general processing on mouse button events and keyboard events. It demonstrates how input messages are received, and what additional information comes with the message. When run, it displays the current pointer location, whether the right or left mousebutton was clicked, and outputs a message based on a 5 second timer. maskblt: The maskblt sample is intended to demonstrate masked bit block transfers. It is unclear that it is working correctly, either in its current form, or as received from Microsoft. However, it does demonstrate basic calls to functions that support bitmaps and keyboard accelerators used by clipboard functions. MDI: The MDI sample demonstrates the multiple document interface. It creates a simple window that can create child windows within itself. It also can set the titles on these child windows. menu: The menu sample demonstrates the use of popup menus, user defined menus and menu functions. It is a very simple program that allows you to select items off a menu, invoke a pop-up menu, and basically do little else, but is a good example of how to start with menus. mltithrd: The mltitrhd sample demonstrates the use of multiple threads. The main routine, mltithrd.f90, has a brief guide to writing multi-threaded applications at the top of the file. It uses the multi-document interface to create child windows, and bounces colored traces through each one. output: The output sample demonstrates using TextOut() and DrawText() to write to a window. It is a very simple sample, outputting text and a few
Page 47 of 58
simple shapes, such as a rectangle, ellipse, and pie-wedge. owncombo: The owncombo sample demonstrates the use of functions and messages for combo boxes and owner-draw control styles. It brings up a window and lets you play with the different combo box styles. paths: The paths sample demonstrates the use of paths and clipping regions. A path is one or more figures or shapes that are filled, outlined, or both filled and outlined. Paths are used in various drawing and painting applications. This sample demonstrates how 6 different types of paths can be set up. With this release, when the application first runs, the window is too small for the titles on the paths, and must be manually resized. platform: The platform sample demonstrates some very simple menu choices, and the popup dialog boxes that you can create from them. This sample displays "menus" that give you information about which platform you are running on. plgdraw: The plgdraw sample demonstrates the PlgBlt() API. The PlgBlt function performs a bit-block transfer of the bits of color data from the specified rectangle in the source device context to the specified parallelogram in the destination device context. The sample displays a rendered version of the contents of the middle grid in the leftmost grid. polydraw: The polydraw sample demonstrates capturing mouse clicks and movement, and combining that with some keyboard input (the Ctrl or Shift keys). It is a very simple program that allows the user to draw lines, curves, and move some points around on the screen. process: The process sample demonstrates creating and terminating processes from within another program. The user interface is very simple, a dialog box with two buttons "Create" and "Terminate", and window with the list of created processes. registry: The registry sample demonstrates the Registry API. This sample displays information from the system registry, allowing the user to traverse the various trees. It is a fairly simple sample, but could be used as an example. select: The select sample demonstrates how to manipulate a cursor and select a region. This sample has two projects under it: 1. Select, which provides a dynamic link library (dll) of routines for selecting a region 2. Demo, which contains an executable program that uses the select library above. To use this sample, first set select as the current project and build select.dll, then set demo as the current project to build the executable. setinfo: The setinfo sample demonstrates how to set the attributes of a file. It has a simple interface, allowing the user to enter a filename in an edit field. The GetInfo button will retrieve information about the file and display it. To set file information, modify the values in the Time and Date edit fields, and click the SetInfo button. To set file attributes, set the appropriate check boxes and click on the SetAttr button. This sample also demonstrates the use of the GetFileTime() and
Page 48 of 58
SetFileTime() API calls. sharemem: The sharemem sample demonstrates the use of named shared memory between two independent processes. There are two projects in this sample, sharemem and othrproc. These can be built in either order; neither depends on the other. To use this sample, first start Sharemem, then start Othrproc. (Note, this is best done outside of Developer Studio.) The visual effect is better if the focus remains with Othrproc, while the mouse moves with Sharemem. A "shadow" mouse is displayed on Othrproc with the same coordinates as the mouse on Sharemem. startp: The startp sample demonstrates various CreateProcess() parameters, including starting processes in a given priority class. This is very similar to the "start" command but with added functionality. It is best run from outside Developer Studio, but when run from within Developer Studio, will give usage information in a console window. streblt: The streblt sample demonstrates the use of the StretchBlt function. This Block Transfer function goes beyond a simple BitBlt, giving the option of stretching or compressing a bitmap to fit a different area in the destination device context. subclass: The subclass sample demonstrates subclassing of standard windows controls, such as button, edit field, and list box. It allows the user to create an arbitrary number of child controls on the main window. These controls are subclassed, and the subclass procedure provides the user a way to move and size the controls. A menu item switches in and out of "Test Mode". When this is on, the subclass procedure passes all message through to the standard procedure, and the controls act just like normal. takeown: The takeown sample demonstrates how to give an Administrator access to a file that has been denied to all. It must be run from an Administrator account. There is a sample file provided named takeown.txt which may be used by the sample. termproc: The termproc sample demonstrates how to terminate a process, given its process id. Although this can be built within the Developer Studio, it is best run from the command line. threads: The threads sample demonstrates suspending, resuming, and setting the priorities of threads. The display shows two threads, red and green, which draw paths across the window. There are menu items to suspend, resume, and set the priority of the two threads. Caution, if the priority of one or the other of the threads is set too high little else will run on the system. timers: The timer sample demonstrates the use of the SetTimer() and KillTimer() APIs. It creates 4 rectangles on its window, each with a different and inverts the color whenever the timer goes off. This sample demonstrates two timer mechanisms. tls: The tls sample demonstrates multiple threads calling into a single DLL. It also demonstrates the use of DLLEXPORT to export entry points from a DLL, rather than using .DEF files. The tls sample has two projects, tlsdll and tls. The workspace is configured such that the tls project is dependent
Page 49 of 58
on the tlsdll project. If you build tls, it will automatically build tlsdll. When running, each thread is blocked waiting for the user to click OK in a message box. The thread may be running in a minimized window. virtmem: The virtmem sample demonstrates the use of various virtual memory APIs. The sample has a main window with menu items that will modify a set page to have different attributes. In order to write to memory, the page must be committed and set to read/write access. Attempting to write to a read only page will cause an access error, "memory cannot be read". world: The world sample demonstrates scaling and translating an image from a metafile with world coordinate transforms. It uses the calls SetWorldTransform() and PlayEnhMetaFile() to do the linear transformation between world space and page space, and to play the enhanced metafile. The sample provides a single sample metafile, sample.emf, although others could be used. wxform: The wxform sample demonstrates world transforms. It has three dialog boxes that display help, track mouse movements, or allow the user to enter transform information. The different quadrants of the screen have different properties, as described in the help box. This sample also demonstrates using messages to communicate between different window procedures.
CelsiCon: The CelsiCon sample is a Fortran Console application that uses a dialog box. It uses an undocumented hook into the dialog initialization event WM_INITDIALOG. It also sets special customizations on that dialog box by using Win32 calls. With a simple modification to the source program, it can also be built as a QuickWin application. dllprgrs: The dllprgrs sample demonstrates using the Visual Fortran Dialog Procedures in a DLL. The public routines in the DLL can be called to: DisplayProgressBox Display a dialog box containing a progress bar SetProgress Set the current progress bar value DestroyProgressBox Destroy the dialog box fxplorer: The fxplorer sample demonstrates an SDI Fortran Windows application using the Visual Fortran Dialog Procedures. It uses an Internet Explorer ActiveX control in the main window of the application. The Internet Explorer control is installed with Internet Explorer. If Internet Explorer is not installed, this sample will not run. mmplayer: The mmplayer sample demonstrates a dialog-based Fortran Windows application using the Visual Fortran Dialog Procedures. It uses an ActiveMovie control in the main window of the application. The ActiveMovie
Page 50 of 58
control is installed with Internet Explorer. If Internet Explorer is not installed, this sample will not run. progress: The progress sample program demonstrates a "dialog-based" Fortran application using the Visual Fortran Dialog Routines, including the progress bar. The main window of the application is a dialog box. It contains Start, Stop, and Exit buttons plus a progress bar. The Start button begins a lengthy operation that is broken into 100 steps. In this simple example, each step is sleeping for .25 seconds. showfont: The showfont sample demonstrates a "dialog-based" Fortran application using the Visual Fortran Dialog Routines. The main window of the application is a dialog box. This program demonstrates the use of a Tab control. The Tab control is used to select the properties of a font. The font is used to display a sample text string. therm: The therm program demonstrates a "dialog-based" Fortran application using various Visual Fortran Dialog Routines, including several that are new to CVF V6.0. The main window of the application is a dialog box. This program uses a Spin control, a Slider control, a Progress bar control, and Picture controls. It also demonstrates using multiple modeless dialog boxes. whizzy: The whizzy sample demonstrates many of the controls available with the dialog manager module provided by Visual Fortran. Note that it is built with the "Logical Values" compatibility switch set. This is required for correct interaction with the dialog manager routines.
DLLEXP1: The DLLEXP1 sample demonstrates a workspace that contains two projects: one that creates a DLL and a second that uses that DLL, linking against the import library created by the first project. Note that for the Fortran Console application to run, the DLL created by the first project must be found somewhere along the PATH. DLLEXP2: The DLLEXP2 sample demonstrates how COMMON variables defined in a DLL can be shared between multiple programs. Two programs are built against the same DLL. The first program is run, and then PAUSEs. The second program is run and changes the COMMON variables, and exits. When the first program resumes, it displays the changed values.
Page 51 of 58
LoadExp1: The LoadExp1 sample demonstrates how to dynamically map a DLL without using an import library. It is a workspace with two projects, one the DLL and the other the executable that uses it.
ClearFP: The ClearFP sample demonstrates CLEARSTATUSFPQQ, a subroutine to clear the ia32 floating-point status word exception flags. Cslexcp2: The Cslexcp2 sample demonstrates exception and termination handling in a Fortran Console application. Cslexcp4: The Cslexcp4 sample demonstrates how to wrap a Fortran entry point inside a C try/except block to allow testing for exceptions that might occur in the protected Fortran code. GetEptrs: The GetEptrs sample demonstrates how to establish a signal handler with SIGNALQQ (or the C signal() function) and use the TRACEBACKQQ and GETEXCEPTIONPTRSQQ Fortran library routines to generate a traceback display from the handler. Winexcp1: The Winexcp1 sample demonstrates exception handling in a Fortran Windows application. It was based on the GENERIC sample, with structured exception handling (SEH) incorporated. VBVF1: The VBVF1 sample demonstrates exception handling in a Visual Basic application that calls a Visual Fortran DLL. This is an ia32-only sample.
unittest: The unittest sample contains 80 small functions that test each of the features of the ISO varying string module. Some of the features tested are concatenation, assignment, and searching for a substring. vocab: The vocab sample uses the ISO varying string module to read a text file and keep a list of each word found in the file and its frequency. When the file is completely read, it prints out the word list and the frequency of each word. wrdcnt: The wrdcnt sample uses the ISO varying string module to read a text file and keep count of how many words are seen. When the file is completely read, it prints out the total number of words found in the file.
Page 52 of 58
Miscellaneous (Misc)
There are a number of miscellaneous samples that demonstrate some specific aspects of Visual Fortran or its utilities. The following miscellaneous examples can be found in separate subdirectories under ...\Df98\Samples\Misc:
l
calcaver: The calcaver sample demonstrates Fortran 90 features, including an interface block, assumed shape array, and the use of modules. It does have a Developer Studio workspace associated with it, or can be built using nmake and the makefile. char2: The char2 sample demonstrates the use of the utility fsplit90. This utility creates separate files for each program unit within a file. Once it has been split, it is identical to charexmp. Build this sample using nmake. charexmp: The charexmp sample does some simple character handling, generating arrays of the alphabet and outputting them to the screen. It has a Developer Studio workspace associated with it, or it can be built using nmake. console: The console sample demonstrates some of the basic consolemanipulating routines. Console sets the size of the console, clears the console, and positions the cursor to write a line at the top of the console, and again at a line further below. A DOS command prompt is an example of a console. cvt_lf_unf: The cvt_lf_unf sample converts data files created with Lahey Fortran to an unformatted binary file. There are two projects within this sample: cvt_lf_unf which builds the conversion utility test which uses cvt_lf_unf to convert a data file created by Lahey Fortran and then read it from CVF. fctrl: The fctrl sample is a very simple one, demonstrating recursion. It is a console application, and computes and displays the factorials of the integers 1 to 12. 13! is too big for a 32-bit integer. This sample can be built either from within Developer Studio, or by using the makefile. fivt: The fivt sample is a basic Installation Verification program for Visual Fortran. It compiles and links a program, and then reads the listing file to make sure the expected version of the compiler was used. It can be compiled and run from either the command line or the Developer Studio. To run fivt from the command line, either set FORT5=FIVT.INP, or invoke it as FIVT < FIVT.INP. To run from Developer Studio, refer to the README.TXT file in that directory. forprint: The forprint sample is a console application that uses the OpenFile and Print dialog boxes to select and print text files, including files with Fortran carriage control. Forprint preprocesses the file, generating a text file that can be printed by any printer. This sample can be built either from within Developer Studio or by using the makefile. fppdemo: The fppdemo sample demonstrates the use of the Fortran Preprocessor, FPP. FPP is a C-like preprocessor, as opposed to the
Page 53 of 58
preprocessing directives that are part of the Fortran 90 standard. It is easiest to build using the makefile, but there is a Developer Studio workspace for it. This workspace has 5 projects, one per variation of fpp defines. It illustrates how to pass fpp flags to the compiler from the Developer Studio; it must be done by hand from within the Settings dialog box. mandel: The mandel sample demonstrates a simple Win32 application with a C main program calling a Fortran subroutine. uopen: The uopen sample demonstrates the use of the USEROPEN keyword from the OPEN statement. In this particular example, the Fortran file calls a C program to do the specialized OPEN. lcinfo: The lcinfo sample demonstrates the use of NLS routines to support some natural language extensions. It also demonstrates how the string order is affected by choice of natural language. The lcinfo sample is a console application that prints out information about what the different language and currency settings are on the computer.
Mixed-Language (Mixlang)
Visual Fortran can be called by a number of other languages, including Visual C/C++, Visual Basic, and MASM (Assembler). Also, it can call other languages, such as C/C++. The following samples in subdirectories under ...\Df98 \Samples\Mixlang demonstrate some of the calling standards or argument definitions that must be used to effectively interwork with other languages:
l
c_call_f: The C_Call_F sample demonstrates a mixed-language application in which a C (or C++) main program calls a Fortran subroutine. It is presented in two different styles; one where the Fortran and C code are built together in the same project and one where the Fortran code is built separately into a DLL which is then called from C. This sample also demonstrates how to pass character/string arguments to Fortran. delphi: This ia32-only sample demonstrates various interactions between Visual Fortran and BorlandTM Delphi. Delphi must be installed to build and run this sample: CallDVF: The CallDVF sample demonstrates calling Fortran from Delphi. There are examples of an integer function, a REAL function, and a character function. spline: The spline sample demonstrates mixed language programming with Visual C/C++. Spline is easiest built from within Developer Studio. The project contains four projects, Console, QuickWin, VC, and spldll. The last project, spldll, creates the dll file used by the other projects. The spline sample implements a "best fit curve" algorithm to a set of coordinates. The coordinates are input by clicking on a window or through a data file. Each of the first three projects within this workspace create a different program, using different paradigms: Console creates an application that runs in a console window and
Page 54 of 58
takes input from a data file, fsplit.dat QuickWin creates an application using the QuickWin interface that takes input from mouse clicks. VC creates an application using Visual C/C++ and the Microsoft Foundation Classes that takes input from mouse clicks.
This workspace and its projects also demonstrate one method of dealing with interproject dependencies. It copies the dependent .DLL file to the local directory before linking and running the project. vb: This set of samples demonstrates various interactions between Visual Fortran and Visual Basic (VB). In all cases, there is an executable that is the VB program which you can run. There is also the VB Project, as well as the Visual Fortran project available so you can modify them as you wish: arrays: The arrays sample demonstrates how to pass arrays of strings from VB to CVF callback: The callback sample demonstrates how to invoke VB callback routines passed to a CVF subroutine returnstring: The returnstring sample demonstrates how to call a Fortran character string function from Visual Basic. simple: The simple sample shows very basic interactions between Visual Fortran and VB, passing numbers from one to the other typearrays: The typearrays sample demonstrates how to pass an array of user-defined types from VB to Visual Fortran, including how to write into some fields of that type from within the Fortran routine. MASM: This set of ia32-only samples demonstrates various interactions between Visual Fortran and MASM, Microsoft's Assembly Language. MASM must be installed to build these samples: simple: The simple sample demonstrates the most simple call from Fortran to MASM. vecprod: The vecprod sample demonstrates how to calculate the inner product of two vectors in a MASM program. These two vectors were passed from Fortran to MASM, and the result was passed back to the Fortran main program. allocabl: The allocabl sample is a rewrite of vecprod using allocatable arrays rather than static ones.
QuickWin (Quickwin)
Visual Fortran provides a library of routines called QuickWin. QuickWin is a library that lets you build applications with a simplified version of the Windows interface with Visual Fortran. The QuickWin library provides a rich set of Windows features, but it does not include the complete Windows Applications Programming Interface (API). If you need additional capabilities, you must set up a Windows application to call the Win32 API directly rather than using QuickWin to build your program. The following samples demonstrate how to use QuickWin, and how to program around some of the strange paradigms of the package.
Page 55 of 58
The following QuickWin examples can be found in subdirectories under ...\Df98 \Samples\QuickWin:
l
calendar: The calendar sample is a simple QuickWin application. It displays a graphical monthly calendar for the current month. chaos: The chaos sample is a QuickWin application that demonstrates the use of menus, dialog boxes for input and output, message boxes for the signaling of error states, various ways of recovering from error conditions and still keep the program running, QuickWin graphics routines, multithreading, and Fortran 90. There is a document in the CHAOS subdirectory, README.TXT, that describes this program and its algorithms in more detail. cleanwin: This sample illustrates how to use native Windows APIs to radically change the appearance of a standard QuickWin window. conapp: This sample demonstrates a method for creating a console window from within a QuickWin application. The sample has two parts: a DLL which creates the console window and performs I/O to or from it, and a QuickWin program that calls that DLL. To allow the DLL to access the console using units *, 5, and 6, this sample uses a technique not recommended for any other use. The DLL is linked against the Fortran dynamic-link libraries, but the QuickWin application is linked against the Fortran static libraries. Generally this is discouraged, because using the two different libraries means that the main program and the DLL cannot share I/O units. In this case, that is the intended outcome. dirkeys: This sample demonstrates a mechanism for passing directional keys from QuickWin to the user's program. dirkeys2: This sample illustrates the use of a callback procedure (hook) using SetWindowsHookEx for keyboard events. It demonstrates how to get around the limitations of the PASSDIRKEYS routine, specifically how to trap for Insert, Delete, and Ctrl-C keys. dirkeys4: This sample demonstrates the PASSDIRKEYSQQ routine. The PASSDIRKEYSQQ routine allows easy access to the Insert, Delete, and CtrlC keys. menudriv: This sample illustrates three programming idioms for use in QuickWin applications. These are: using the menu to drive various functions; getting control on EXIT; using the status bar to indicate program progress. peekapp: This multi-threaded sample simulates a compute loop with Peekchar and Getchar to handle input as it comes in character by character. It is intended to be used within a QuickWin program. peekapp3: This multi-threaded sample is similar to peekapp, but contains a procedure to perform an orderly shutdown on the character-reading thread. poker: The poker sample demonstrates the use of QuickWin and some simple controls on the window. This sample uses QuickWin to manage the windowing interface, but uses some of the Win32 API calls, and must be linked against dfwin.lib. It should be mentioned that the file CARDS.F90 takes a longer time to compile than any of the o ther files that make up the
Page 56 of 58
poker sample. Also, before building, you should make sure you update all dependencies, as there are many. postmini: The Postmini graphical postprocessing software can be used to examine the output of a number of process, device and circuit simulation (TCAD) programs, as well as import data from ASCII files. This version of PostMini uses QuickWin for the graphical interface. Other versions are available for Compaq Tru64 UNIX and OpenVMS Alpha that use other graphical interfaces. Refer to the README file in the PostMini directory for details on this sample. qwpaint: The qwpaint sample demonstrates using mouse callbacks and graphics routines in a QuickWin application. It has two windows, a "control grid" window and a "canvas" window. QuickWin's built-in File.Save option can be used to save the contents of either window as a bitmap (.BMP) file. qwpiano: The qwpiano sample demonstrates using mouse callbacks and menu callbacks in a QuickWin application. It draws a graphical piano keyboard on the screen that can be played using the mouse. It should be noted that two compatibility options, "Logical Values" and "Filenames from Command Line" are set for this sample. The former affects interacting with the QuickWin routines, and the latter causes a File dialog box to come up in response to an OPEN request. Resize: The resize sample demonstrates how to use Win32 API calls from within a QuickWin application to correctly size the information to be displayed to a QuickWin window. testscrl: The testscrl sample demonstrates how to scroll to the last line of a window that has scroll bars. It also shows how to make a window without scroll bars.
scigraph: The SciGraph sample offers a starting scientific graphing package to support some basic forms of graphs. There are two projects in this sample: Scigraph, which creates a library and some .MOD files Sgdemo, which creates a program to demonstrate some of the basic forms of graphs Build the Scigraph project first, followed by Sgdemo. Note that the output area for the SCIGRAPH.LIB project is included as a path to find .MOD files. There is a .WRI file in the SCIGRAPH directory that contains basic reference
Page 57 of 58
information on the various SciGraph calls, including a list of error returns and possible causes. A known limitation of SciGraph is directly related to the known QuickWin limitation of 256 colors on the graphics device. This limitation shows itself when trying to print or save as a bitmap file; only a black screen is printed or saved.
Tutorial
All Tutorial samples are simple programs, and most of them are standalone. Each contains compilation information at the top of the file, both for command line and from within Developer Studio. If there are multiple files, this is noted at the top of the program. All tutorial samples can be found in the directory ...\Df98\Samples\Tutorial:
l
l l
assocn: The assocn sample illustrates the non-associativity of floatingpoint addition. cycle: The cycle sample illustrates the Fortran 90 statements CYCLE and EXIT. derived: The derived sample demonstrates a derived type structure, type variables, and assign values. dtypearg: The dtypearg sample demonstrates the use of SEQUENCE to allow passing a derived-type argument. dtypecom: The dtypecom sample demonstrates placing derived-type objects in COMMON blocks. dtypemod: The dtypemod sample demonstrates the use of a module to allow passing a derived-type argument. It also shows WRITE of a derivedtype. epsilon: The epsilon sample illustrates several calculations of machine epsilon. internal: The internal sample illustrates the use of an internal subroutine and the CONTAINS statement. interval: The interval sample demonstrates the use of rounding control. This sample must be compiled without optimizations. keynames: The keynames sample demonstrates the Fortran concept of using a type structure and a subroutine defined within a module. This sample also requires NAME_SSN.F90 to build. matherr: The matherr sample demonstrates handling run-time math exceptions. This sample is not a standalone application, but can be included in other programs. mathtest: The mathtest sample uses matherr above to demonstrate the response to error conditions on intrinsic math calls. It uses the MATHERRQQ subroutine in MATHERR.F90. mbcomp: The mbcomp sample demonstrates the use of the National Language Support (NLS) routines. namelist: The namelist sample demonstrates the NAMELIST Statement. percent: The percent sample demonstrates a user-defined operator, in
Page 58 of 58
this case a unary operator .c. that converts a floating point number to a "percentage". pointer: The pointer sample demonstrates allocating and deallocating pointers and targets. pointer2: The pointer2 sample demonstrates pointing at a pointer and target, and the use of the ASSOCIATED intrinsic. realg: The realg sample demonstrates using coordinate graphics. This sample uses the QuickWin package to display to the screen. recurs: The recurs sample is a very simple demonstration of a recursive subroutine. showfont: The showfont sample uses QuickWin to display a variety of fonts. This sample demonstrates various calls to QuickWin. sigtest: The sigtest sample demonstrates user-defined exception handlers. It shows how to establish the name of the exception handler as the function to be invoked if an exception happens. sine: The sine sample demonstrates the use of QuickWin to output sine waves and some simple shapes. testget: The testget sample illustrates the use of internal READ to convert from REAL to CHARACTER. This is best run from outside of Developer Studio, and can be built simply by df testget.f90. threads: The threads sample demonstrates multiple threads with QuickWin. It is a much simpler program than advanced\win32\bounce, but is not as efficient at run time. unclear: The unclear sample demonstrates the use of generic module procedures. A generic procedure called "dup" has three instances; one with real arguments, one with integer arguments, and one with character arguments. where: The where sample demonstrates the Fortran 90 constructs WHERE and ELSEWHERE.
Page 1 of 19
The supplied Fortran Command Prompt window in the Compaq Visual Fortran program folder, in which the appropriate environment variables in DFVARS.BAT are preset. Your own terminal window, in which you have set the appropriate environment variables by executing the DFVARS.BAT file. This file sets the environment variables (such as PATH, INCLUDE, and LIB) and allows use of IMSL and CXML libraries. By default, DFVARS.BAT is installed in \Program Files\Microsoft Visual Studio\Df98\BIN.
Calls the Visual Fortran compiler to process Fortran files. Passes the linker options to the linker. Passes object files created by the compiler to the linker. Passes libraries to the linker. Calls the linker or librarian to create the .EXE or library file.
The DF command automatically references the appropriate Visual Fortran RunTime Libraries when it invokes the linker. Therefore, to link one or more object files created by the Visual Fortran compiler, you should use the DF command instead of the LINK command. Because the DF driver calls other software components, error messages may be returned by these other components. For instance, the linker may return a message if it cannot resolve a global reference. Using the /watch:cmd option on the DF command line can help clarify which component is generating the error. This section contains the following information:
l l l l l
The Format of the DF Command Examples of the DF Command Format Input and Output Files Environment Variables Used with the DF Command Specifying Project Types with DF Command Options
Page 2 of 19
Redirecting Command-Line Output to Files Using the DF Command to Compile and Link DF Indirect Command File Use Compiler and Linker Messages
Except for the linker options, options can be specified in any order. Linker options must be preceded by the keyword /link and must be specified at the end of the command line, following all other options.
options A list of compiler options (or linker options preceded by the /link option). These lists of options take the following form:
[/option:[arg]] [filename.ext]...
Where: /option[:arg] Indicates either special actions to be performed by the compiler or linker, or special properties of input or output files. The following rules apply to options and their names:
l
Options begin with a slash (/). You can use a dash (-) instead of a slash. Visual Fortran options are not case-sensitive. Certain options provided for compatibility with Microsoft Fortran PowerStation options are casesensitive, such as /FA and /Fafile. You can abbreviate option names. You need only enter as many characters as are needed to uniquely identify the option.
Page 3 of 19
option name. For example, the /warn option accepts several keywords, including argument_checking and declarations. To specify only a single keyword, specify the keyword after the colon (:). For example, the following specifies the /warn option declarations keyword:
DF /warn:declarations test.f90
To specify multiple keywords, specify the option name once, and place each keyword in a comma-separated list enclosed within parentheses with no spaces between keywords, as follows:
DF /warn:(argument_checking,declarations) test.f90
filename.ext Specifies the files to be processed. You can use wildcard characters (such as *.f90) to indicate multiple files or you can specify each file name. The file extension identifies the type of the file. With Fortran source files, certain file extensions indicate whether that source file contains source code in free (such as .f90) or fixed (such as .for) source form. You can also specify compiler options to indicate fixed or free source form (see /[no] free). The file extension determines whether that file gets passed to the compiler or to the linker. For example, files myfile.for and projfile.f are passed to the compiler and file myobj.obj is passed to the linker. Compiler Options, Alphabetic List The following table lists the DF command compiler options alphabetically: /[no]alignment /architecture /[no]asmfile /[no]automatic /[no]browser /[no]check /[no]altparam /[no]asmattributes /assume /bintext /ccdefault /[no]comments
Using the Compiler and Linker from the Command Line /[no]compile_only /cxml /[no]dbglibs /define /[no]error_limit /[no]extend_source /extfpp /[no]f66 /fast /[no]fltconsistency (ia32 only) /fpe /[no]fpscomp /help or /? /imsl /[no]inline /integer_size /[no]libdir /[no]link /[no]logo /[no]map /[no]module /[no]object /[no]pad_source /[no]pipeline (ia64 only) /real_size /convert /[no]d_lines /[no]debug /dll /[no]exe /extfor /extlnk /[no]f77rtl /[no]fixed /[no]fpconstant /fpp /[no]free /iface /[no]include /[no]intconstant /[no]keep /libs /[no]list /[no]machine_code /math_library /names /[no]optimize /[no]pdbfile /preprocess_only /[no]recursive
Page 4 of 19
Using the Compiler and Linker from the Command Line /[no]reentrancy /source /[no]stand /[no]threads /[no]transform_loops /undefine /[no]vms /[no]watch /winapp For more information:
l l l
Page 5 of 19
On DF command examples, see Examples of the DF Command Format On compiler option categories, see Categories of Compiler Options On using the FL32 command, see Microsoft Fortran PowerStation Command-Line Compatibility About Fortran PowerStation options (such as /MD) and their DF command equivalents, see Equivalent Visual Fortran Compiler Options
In this example, the .f90 file extension indicates test.f90 is a Fortran freeform source file to be compiled. The file extension of obj indicates proj.obj is an object file to be passed to the linker. You can optionally add the /link option before the file proj.obj to indicate it should be passed directly to the linker. In the following example, the /check:bounds option requests that the Fortran compiler generate additional code to perform run-time checking for out-of-bounds array and substring references for the files myfile.for and
Using the Compiler and Linker from the Command Line test.for (fixed-form source):
DF /check:bounds myfile.for test.for
Page 6 of 19
In the following example, the /link option indicates that files and options after the /link option are passed directly to the linker:
DF myfile.for /link myobject.obj /out:myprog.exe
Invalid DF commands The following DF command is invalid because the /link option indicates that items after the /link option are passed directly to the linker, but the file test.for should be passed to the compiler:
DF myfile.for /link test.for /out:myprog.exe
The following DF command is invalid because the /link option is missing and the /out linker option is not recognized as a compiler option:
DF myfile.for test.for /out:myprog.exe
In this case, you can alternatively use one of the DF options (/exe) that specifies information to the linker:
DF myfile.for test.for /exe:myprog.exe
Environment Variables Used with the DF Command Specifying Project Types with DF Command Options Using the DF Command to Compile and Link DF Indirect Command File Use Compiler and Linker Messages
Page 7 of 19
the linker. The following types of files are used with the DF command:
l
Files passed to the compiler: .f90, .for, .f, .fpp, .i, .i90, .inc, .fi, .fd, .f77 Typical Fortran (DF command) source files have a file extension of .f90, .for, and .f. When editing your source files, you need to choose the source form, either free-source form or fixed-source form (or a variant of fixed form called tab form). You can either use a compiler option to specify the source form used by the source files (see /fixed or /free), or you can use the following file extensions when creating or renaming your files:
The compiler assumes that files with an extension of .f90, .F90, or .i90 are free-form source files. The compiler assumes that files with an extension of .f, .for, .FOR, or .i are fixed-form (or tab-form) files.
Files passed to the linker: .lib, .obj, .o, .exe, .res, .rbj, .def, .dll For example, object files usually have a file extension of .obj. Files with extensions of .lib are usually library files.
l l
l l
An object file (.OBJ) if you specify the /compile_only, /keep, or /object option on the command line. An executable file (.EXE) if you do not specify the /compile_only option A dynamic-link library file (.DLL) if you specify the /dll option and do not specify the /compile_only option A module file (.MOD) if a source file being compiled defines a Fortran 90 module (MODULE statement) A program database file (.PDB) if you specify the /pdbfile or /debug:full (or equivalent) options A listing file (.LST) if you specify the /list option A browser file (.SBR) if you specify the /browser option
You control the production of these files by specifying the appropriate options on the DF command line. Unless you specify the /compile_only option or /keep option, the compiler generates a single temporary object file from one or more source files. The linker is then invoked to link the object file into one executable image file. If fatal errors are encountered during compilation, or if you specify certain options such as /compile_only, linking does not occur. When a path or file name includes an embedded space or a special character, enclose the entire file location name in double quotation marks ("). For example:
Page 8 of 19
DF "Project xyz\fortmain.f90"
For more information about naming input and output files, see:
l l
The /exe:file or the /out:file linker option to name an executable program file. The /dll:file alone or the /dll option with the /out:file linker option to name an executable dynamic-link library.
You can also use the /object:file option to specify the object file name. If you specify the /compile_only option and omit the /object:file option, each source file is compiled into a separate object file. For more information about the output file(s) created by compiling and linking multiple files, see Compiling and Linking Multiple Fortran Source Files. Many compiler options allow you to specify the name of the file being created. If you specify only a filename without an extension, a default extension is added for the file being created, as summarized below: Default File Extensions for Compiler Options Option /asmfile:file /browser:file /dll:file /exe:file /list:file /map:file /pdbfile:file Default File Extension .ASM .SBR .DLL .EXE .LST .MAP .PDB (default filename is df60.pdb)
Temporary Files
Page 9 of 19
Temporary files created by the compiler or linker reside in the directory used by the operating system to store temporary files. To store temporary files, the operating system first checks for the TMP environment variable. If the TMP environment variable is defined, the directory that it points to is used for temporary files. If the TMP environment variable is not defined, the operating system checks for the TEMP environment variable. If the TEMP environment variable is not defined, the current working directory is used for temporary files. Temporary files are usually deleted, unless the /keep option was specified. For performance reasons, use a local drive (rather than using a network drive) to contain the temporary files. To view the file name and directory where each temporary file is created, use the /watch:cmd option. To create object files in your current working directory, use the /compile_only or /keep option. Any object files (.obj files) that you specify on the DF command line are retained.
Description The PATH environment variable sets the search path. The linker uses the LIB environment variable to determine the location of .LIB files. If the LIB environment variable is not set, the linker looks for .LIB files in the current directory. The LINK_F90 environment variable contains a list of libraries used for linking IMSL libraries (Professional and Enterprise Editions), as listed in Library Naming Conventions. The make facility (NMAKE) uses the INCLUDE environment variable to locate INCLUDE files and module files. The Visual Fortran compiler uses the INCLUDE environment variable to locate files included by an INCLUDE statement or module files referenced by a USE statement. Similarly, the resource compiler uses the INCLUDE environment variable to locate #include and RCINCLUDE files.
LINK_F90
INCLUDE
Page 10 of 19
The DF environment variable can be used to specify frequently used DF options and files. The options and files specified by the DF environment variable are added to the DF command; they are processed before any options specified on the command line. You can override an option specified in the DF environment variable by specifying an option on the command line. For information about using the DF environment variable to specify frequently-used options, see Using the DF Environment Variable to Specify Options.
You can set these environment variables by using the DFVARS.BAT file or the Fortran Command Prompt command-line window (see "Using the Command-Line Interface" in Compaq Visual Fortran Installing and Getting Started). For a list of environment variables recognized at run-time, see Run-Time Environment Variables.
To create a Fortran Console Application with the DF command, you do not need to specify any options (if you link separately, specify the link option /subsystem:console). This is the default project type created. To create a Fortran Standard Graphics Application with the DF command, specify the /libs=qwins option (which also sets certain linker options). To create a Fortran QuickWin Application with the DF command, specify the /libs:qwin option (which also sets certain linker options). To create a Fortran Windows Application with the DF command, specify the /winapp option (which also sets certain linker options).
To create a Fortran COM Server, you must use Developer Studio (see Creating COM Servers); you cannot create a Fortran COM Server project type with the DF command. The following types are library projects, without main programs:
l
Using the Compiler and Linker from the Command Line the /dll option (sets the /libs:dll option).
l
Page 11 of 19
If your application will not call any QuickWin or Standard Graphics routines, specify /libs:static and /compile_only options to create the object files. Use the LIB command to create the library (see Managing Libraries with LIB). If your application will call QuickWin routines, specify /libs:qwin and /compile_only options to create the object files. Use the LIB command to create the library (see Managing Libraries with LIB). If your application will call Standard Graphics routines, specify /libs:qwins and /compile_only options to create the object files. Use the LIB command to create the library (see Managing Libraries with LIB).
On Windows NT 4 and Windows 2000 systems, to place standard output into file one.out and standard error into file two.out, type the df command (with its filenames and options) as follows:
df filenames /options 1>one.out 2>two.out
To place standard output and standard error into a single file both.out on a Windows NT 4 or Windows 2000 system, type the df command as follows:
df filenames /options 1>both.out 2>&1
On Windows 98, Windows Me, and Windows 95 systems, use the EC command-line tool to place standard output and standard error into separate files or into a single file. The EC tool is located on the Visual Fortran CD-ROM in the x86\Usupport\Misc\Win95 folder. Copy ec.exe into a
Page 12 of 19
folder in your command-line path (PATH environment variable) on your hard disk. Precede the DF command line with the EC command and place the entire DF command in quotes. For example, to place standard output into the file one.out and standard error into file two.out, type the EC and DF command (with its filenames and options) as follows:
ec "df filenames /options 1>one.out 2>two.out"
To place standard output and standard error into a single file both.out on a Windows 98, Windows Me, or Windows 95 system, type the EC and DF commands as follows:
ec "df filenames /options 1>both.out 2>&1"
Compiling and Linking a Single Source File Using the DF Environment Variable to Specify Options Compiling, but not Linking, a Fortran Source File Compiling and Linking Multiple Fortran Source Files Generating a Listing File Linking Against Additional Libraries Linking Object Files Compiling and Linking for Debugging Compiling and Linking for Optimization Compiling and Linking Mixed-Language Programs
Alternatively, you can name the executable file by using the linker /out option:
Page 13 of 19
You can also specify additional options on the DF command line. In the following command sequence, the SET command sets the DF environment variable. The DF options specified are /debug:minimal and /list.
set DF=/debug:minimal /list DF myprog.for /show:map
If the options specified on the command line conflict with the options specified by the DF environment variable, the option specified on the command line takes precedence. In the following command sequence, the /debug:minimal option specified by the DF environment variable is overridden by the /debug:none option specified on the command line:
set DF=/debug:minimal /list DF myprog.for /debug:none
If the files a.for, b.for, and c.for were the only .for files in the current directory, you could use a wildcard character to similarly compile the three source files:
DF *.for
Page 14 of 19
If you use the /compile_only option to prevent linking, also use the /object:file option so that multiple sources files are compiled into a single object file, allowing more optimizations to occur:
DF /compile_only /object:a.obj a.for b.for c.for
When you use modules and compile multiple files, compile the source files that define modules before the files that reference the modules (in USE statements). When you use a single DF command, the order in which files are placed on the command line is significant. For example, if the free-form source file moddef.f90 defines the modules referenced by the file projmain.f90, use the following DF command line:
DF moddef.f90 projmain.f90
The following command compiles a.for, b.for, and c.for. It generates three listing files (and three object files) for the three source files:
DF a.for b.for c.for /list /compile_only
The following command sequence compiles and links a.for, b.for, and c.for. It generates one named object file (a.obj) and one listing file (a.lst). The second command links the object files into an executable file (a.exe):
DF a.for b.for c.for /list /compile_only /object:a.obj DF a.obj
The following command sequence compiles and links a.for, b.for, and c.for. It generates three object files (a.obj, b.obj, and c.obj) and three listing files (a.lst, b.lst, and c.lst). The second command links the object files into an executable file (a.exe):
DF a.for b.for c.for /list /compile_only
Page 15 of 19
The following command links against single-threaded DLLs. It links against the default libraries and mylib.lib:
DF x.f90 /libs:dll mylib.lib
To request the creation of a dynamic-link library, see /dll. For more information on the types of libraries available to link against, see the following options:
l l l l l
For more information about compiling and linking Visual Fortran and Visual C++ programs (and the libraries used), see Specifying Consistent Library Types.
Page 16 of 19
DF x.obj
By default, the debugger symbol table information is created in a PDB file, which is needed for the debugger integrated within the visual development environment. If you use separate DF commands to compile and link, you will want to specify the same debugging information level for the compiler and the linker. For example, if you specify /debug:minimal to the compiler, you will also specify /link /debug:minimal. The following command sequence compiles and then links x.for for debugging with the integrated visual development environment debugger:
DF x.for /debug:full /optimize:0 /compile_only DF x.obj /debug:full
For more information about preparing your command-line program for debugging, see Preparing Your Program for Debugging.
Page 17 of 19
If you specify the /compile_only or /keep option without specifying the /object option, each source file is compiled into an object file. This is acceptable if you specified no optimization (/optimize:0) or local optimization (/optimize:1). An information message appears when you specify multiple input files and specify an option that creates multiple object files (such as /compile_only without /object) and specify or imply global optimization (/optimize:2 or higher optimization level). If you specify the /compile_only option, you must link the object file (or files) later by using a separate DF command. You might do this using a makefile processed by the NMAKE command for incremental compilation of a large application. However, keep in mind that either omitting the /compile_only or /keep option or using the /compile_only option with the /object:file option provides the benefit of full interprocedural optimizations for compiling multiple Fortran source files. Other optimization options are summarized in Software Environment and Efficient Compilation.
The cl command (invokes the C compiler) compiles but does not link cfunc.c. The -c option specifies that the linker is not called. This command creates cfunc.obj. The DF command compiles myprog.for and links cfunc.obj with the object file created from myprog.for to create myprog.exe. For more information about compiling and linking Visual Fortran and Visual C++ programs, and the libraries used, see:
l l
Page 18 of 19
The following DF command executes the contents of file text.txt as an indirect command file to create a debugging version of the executable program and its associated PDB file:
DF @test.txt
Indirect command files do not use continuation characters; all lines are appended together as one command.
Compiler Diagnostic Messages and Error Conditions Linker Diagnostic Messages and Error Conditions
The pointer (---^) indicates the exact place on the source program line where the error was found. The following error message shows the format and message text in a listing file when an END DO statement was omitted:
echar.for(7): Severe: Unclosed DO loop or IF block DO I=1,5 --------^
Diagnostic messages usually provide enough information for you to determine the cause of an error and correct it. When using the command line, make sure that the appropriate environment variables have been set by executing the DFVARS.BAT file. These environment variables are preset if you use the Fortran Command Prompt window in the
Page 19 of 19
Compaq Visual Fortran program folder (see "Using the Command-Line Interface" in Compaq Visual Fortran Installing and Getting Started). For example, this BAT file sets the environment variables for the include directory paths. For errors related to INCLUDE and module (USE statement) file use, see /[no] include. For a list of environment variables used by the DF command during compilation, see Environment Variables Used with the DF Command. To control compiler diagnostic messages (such as warning messages), see /[no] warn. To view the passes as they execute on the DF command line, specify /watch:cmd or /watch:all.
Page 1 of 130
Within the visual development environment, by using the Fortran tab in the Project menu, Settings dialog box. The DF command line. Compiler options must precede the /LINK option.
Unless you specify certain options, the DF command line will both compile and link the files you specify. To compile without linking, specify the /compile_only (or equivalent) option. After the /LINK option on the DF command line, you can specify linker options. Linker options and any libraries specified get passed directly to the linker, such as /NODEFAULTLIB. If you choose to use separate compile and link commands, you can also specify linker options on a separate LINK command. This section contains the following information:
l l l
Compiler Options Linker Options and Related information Microsoft Fortran PowerStation Command-Line Compatibility
Compiler Options
This section describes the compiler options and how they are used. It includes the following topics:
l l
Categories of compiler options, according to functional grouping. Descriptions of each compiler option, listed alphabetically.
Compiler and Linker Options shows a sample Fortran tab: Project Settings, Fortran Tab
Page 2 of 130
The options are grouped under functional categories (the initial Category is General, as shown) to help you locate the options needed for your application. From the Fortran tab, you can select one of the following categories from the Category drop-down list (exception: those options listed in the Miscellaneous category below): General Compatibility Debug Floating Point Code Generation Compilation Diagnostics External Procedures Fortran Data
Page 3 of 130
If a compiler option is not available in the dialog boxes, you can enter the option in the lower part of the Project Settings dialog box just as you would at the command line (under Project Options:). The following tables list the Visual Fortran compiler options by category in the Fortran tab: General-Purpose Options Debugging Level Warning Level Optimization Level Predefined Preprocessor Symbols Generate Source Browse Information Code Generation Options Generate Most-Optimized Code Enable Recursive Routines Object text string inserted into object file Math Library: Checking or Fast Performance Generate Code for xxx Chip /[no]fast (changes multiple options) /[no]recursive /bintext:string /math_library /architecture /[no]debug /warn /[no]optimize /define /[no]browser[:file]
Compatibility Options (See also Fortran Data) Unformatted File Conversion (Nonnative Data) Enable VMS Compatibility Enable F77 Run-Time Compatibility Use F77 Integer Constants /convert (also see /assume: [no]byterecl) /[no]vms /[no]f77rtl /[no]intconstant
Compiler and Linker Options Microsoft Fortran PowerStation V4 Compatibility Options Microsoft Fortran PowerStation: Filenames from Command Line Microsoft Fortran PowerStation: I/O Format Microsoft Fortran PowerStation: Libraries Microsoft Fortran PowerStation: List Directed I/O Spacing Microsoft Fortran PowerStation: Logical Values Microsoft Fortran PowerStation: Other Runtime Behavior Microsoft Fortran PowerStation: Predefined Preprocessor Symbols Compilation Diagnostic Options Compilation Error Limit Warning Levels (Ignore, Normal, Treat Warnings as Errors) Fortran Standards Checking (None, Fortran 90, or Fortran 95) Treat Fortran Standard Warnings as Errors Argument Mismatch Data Alignment Inform when Compiling Files Separately (effect on interprocedure optimization) Truncated Source Uncalled Routines Undeclared Variables/Symbols Uninitialized Variables /[no]error_limit
Page 4 of 130 /[no]fpscomp (various keywords listed below) /fpscomp:[no]filesfromcmd /fpscomp:[no]ioformat /fpscomp:[no]libs /fpscomp:[no]ldio_spacing /fpscomp:[no]logicals /fpscomp:[no]general /fpscomp:[no]symbols
/warn:nogeneral, default settings, or /warn:errors /stand:keyword /warn:[no]stderrors /warn:[no]argument_checking /warn:[no]alignments /warn:[no]fileopt /warn:[no]truncated_source /warn:[no]uncalled /warn:[no]declarations /warn:[no]uninitialized
Compiler and Linker Options Unused Variables Usage (Fortran language) Debug Options Debugging Level (None, Minimal, Partial, Full) Compile Lines With D in Column 1 Use Program Database for Debug Information and File Name External Procedures (and Argument Passing) Options Argument Passing Conventions External Names Case Interpretation String Length Argument Passing /[no]iface:keyword /names:keyword /warn:[no]unused /warn:[no]usage
Page 5 of 130
/[no]iface:mixed_str_len_arg
Append Underscore to External Names /assume:[no]underscore Fortran Data Options (See also Compatibility) Default REAL and COMPLEX Kind Default INTEGER and LOGICAL Kind Append Underscore to External Names (under Data Options) Enable Dummy Arguments Sharing Memory Locations Extend Precision of Single-Precision Constants Use Bytes as RECL= Unit for Unformatted Files Variables Default to Automatic or Static Storage Common Element Alignment /real_size:num /integer_size:num /assume:[no]underscore /assume:[no] dummy_aliases /[no]fpconstant /assume:[no]byterecl /[no]automatic or /[no] static /[no]alignment:[no] common
Compiler and Linker Options Structure Element Alignment (Derived Type and Record Data) Allow SEQUENCE Types to be Padded for Alignment Floating-Point Options (See also Optimizations) Floating-Point Exception Handling Enable Floating-Point Consistency Extend Precision of Single-Precision Constants Enable IEEE Minus Zero Support Fortran Language Options Enable FORTRAN 66 Semantics Enable Alternate PARAMETER Syntax Name Case Interpretation Source Form (File Extension, Fixed Form, or Free Form) Fixed-Form Line Length Pad Fixed-Form Source Records Library Options (See also External Procedures) Enable Reentrancy Support Use Run-Time Libraries, DLL version Use Run-Time Libraries, Static library version Use Run-Time Libraries, QuickWin version /[no]f66 /fpe
Page 6 of 130
/alignment:[no]records /alignment:[no]sequence
Use Run-Time Libraries, Standard Graphics version /libs:qwins Use Run-Time Libraries, Debug version /[no]dbglibs
Compiler and Linker Options Use Run-Time Libraries, Multi-Threaded version Disable Default Library Search Rules Disable OBJCOMMENT Library Names in Object Use Common Windows Libraries Use Compaq Extended Math Library (CXML) Use IMSL Math Libraries Listing and Assembly File Options Source Listing Contents of Source Listing File Annotations Assembly Listing /[no]list /[no]threads /libdir:noauto /libdir:nouser /winapp /cxml /imsl
Page 7 of 130
Miscellaneous Linker Tab and Command-Line Options (Not Listed in Fortran Tab) Specify Linker Options (after /link) Generate Link Map Compile, Do Not Link Create Dynamic Link Library (DLL project type) Software Instruction Scheduling Display Help Text File Specify Custom File Extension for Compiler Specify Custom File Extension for Compiler /link (use Linker tab) /[no]map (use Linker tab) /compile_only or /c (use Compile in Build menu) /dll and Specifying Project Types with DF Command Options /[no]pipeline (ia64 only; command line only) /help or /? (command line only) /source (command line only) /extfor (command line only)
Compiler and Linker Options Specify Custom File Extension for Linker Create one object file for each input source file Name of Executable Program or DLL File Name of Object File Perform Syntax Check Only (No Object File) Display Copyright and Compiler Version Display Compilation Details Write C-Style Comments for FPP Specify Custom File Extension for FPP Only Preprocess FPP Files Undefine Preprocessor Symbols
Page 8 of 130
/extlnk (command line only) /[no]keep (command line only) /[no]exe[:file] (command line only) /[no]object[:file] (command line only) /[no]syntax_only (command line only) /nologo and /what (command line only) /[no]watch (command line only) /comments (command line only; for FPP) /extfpp (command line only) /preprocess_only (command line only) /undefine (command line only)
Optimization Options (See also Code Generation) Optimization Level Variables Default to Automatic Storage Enable Dummy Arguments Sharing Memory Locations Transform Loops Enable I/O Buffering Loop Unrolling Math Library: Checking or Fast Performance Inlining Procedures /[no]optimize /[no]automatic /assume:[no]dummy_aliases /[no]transform_loops /assume:[no]buffered_io /unroll /math_library /[no]inline
Compiler and Linker Options Code Tuning for xxx chip Allow Reordering of Floating-Point Operations Software Instruction Scheduling Preprocessor Options Define Preprocessor Symbols Default INCLUDE and USE Path Module path (to place module files) INCLUDE and USE Path Use FPP and specify options Predefined Preprocessor Symbols to FPP Only Run-Time Options Generate Traceback Information Default Output Carriage Control Array and String Bounds Integer Overflow Floating-Point Underflow Power Operations Edit Descriptor Data Type Flawed Pentium Chip Edit Descriptor Data Size /[no]traceback /ccdefault:keyword /check:[no]bounds /check:[no]overflow /check:[no]underflow /check:[no]power /check:[no]format /define /tune
Page 9 of 130
For a table of DF command options listed alphabetically, see Compiler Options, Alphabetic List.
/[no]alignment
Page 10 of 130
Syntax: /alignment[:keyword...], /noalignment, or /Zpn The /alignment option specifies the alignment of data items in common blocks, record structures, and derived-type structures. The /Zpn option specifies the alignment of data items in derived-type or record structures. The /alignment options are:
l
/align:[no]commons The /align:commons option aligns the data items of all COMMON data blocks on natural boundaries up to four bytes. The default is /align:nocommons (unless /fast is specified), which does not align data blocks on natural boundaries. In the visual development environment, specify the Common Element Alignment as 4 in the Fortran Data Compiler Option Category.
/align:dcommons The /align:dcommons option aligns the data items of all COMMON data blocks on natural boundaries up to eight bytes. The default is /align:nocommons (unless /fast is specified), which does not align data blocks on natural boundaries. Specifying /fast sets /align:dcommons. In the visual development environment, specify the Common Element Alignment as 8 in the Fortran Data Compiler Option Category.
/align:[no]records The /align:records option (the default) requests that components of derived types and fields of records be aligned on natural boundaries up to 8 bytes (for derived types with the SEQUENCE statement, see /align:[no]sequence below). The /align:norecords option requests that components and fields be aligned on arbitrary byte boundaries, instead of on natural boundaries up to 8 bytes. In the visual development environment, specify the Structure Element Alignment in the Fortran Data Compiler Option Category.
/align:[no]sequence The /align:sequence option requests that components of derived types with the SEQUENCE statement will obey whatever alignment rules are currently in use (default alignment rules will align unsequenced components on natural boundaries). The default value (unless /fast is specified) is /align:nosequence, which means that components of derived types with the SEQUENCE property will be packed, regardless of whatever alignment
Page 11 of 130
rules are currently in use. Specifying /fast sets /align:sequence. In the visual development environment, specify Allow SEQUENCE Types to be Padded for Alignment in the Fortran Data Compiler Option Category.
l
/align:recNbyte or /Zpn The /align:recNbyte or /Zpn options request that fields of records and components of derived types be aligned on the smaller of:
The size byte boundary (N) specified. The boundary that will naturally align them.
Specifying /align:recNbyte, /Zpn, or /align:[no]records does not affect whether common block fields are naturally aligned or packed. In the visual development environment, specify the Structure Element Alignment in the Fortran Data Compiler Option Category. Specifying /Zp /Zp1 /Zp2 /Zp4 /alignment /noalignment /align:rec1byte /align:rec8byte Is the Same as Specifying /alignment:records or /align:rec8byte /alignment:norecords or /align:rec1byte /align:rec2byte /align:rec4byte /Zp8 with /align:dcommons, /alignment:all, or /alignment:(dcommons,records) /Zp1, /alignment:none, or /alignment: (nocommons,nodcommons,norecords) /align:norecords /align:records
When you omit the /alignment option, records and components of derived types are naturally aligned, but fields in common blocks are packed. This default is equivalent to:
/alignment=(nocommons,nodcommons,records,nosequence)
You can also control the alignment of components in records and derived types and data items in common blocks by Using the cDEC$ OPTIONS Directive.
Page 12 of 130
/[no]altparam
Syntax: /altparam, /noaltparam, /4Yaltparam, or /4Naltparam The /altparam option determines how the compiler will treat the alternate syntax for PARAMETER statements, which is:
PARAMETER par1=exp1 [, par2=exp2] ...
This form does not have parentheses around the assignment of the constant to the parameter name. With this form, the type of the parameter is determined by the type of the expression being assigned to it and not by any implicit typing. In the visual development environment, specify the Enable Alternate PARAMETER Syntax in the Fortran Language Compiler Option Category. When the /[no]altparam or equivalent options are not specified, the compiler default will be to allow the alternate syntax for PARAMETER statements (/altparam). To disallow use of this form, specify /noaltparam or /4Naltparam. To allow use of this form, allow the default or specify /altparam or /4Yaltparam.
/[no]annotations
Syntax: /annotations:keyword or /noannotations The /annotations option specifies that additional information will be added to the source listing file. This information indicates which of a set of optimizations the compiler applied to particular parts of the source file. Note that some of the values of keyword may exist for optimizations that are not supported on your platform. If so, the source listing file contains no corresponding annotations (such as feedback). In the visual development environment, specify the various annotations keywords under the Annotations list box in the Listing Compiler Option Category. You can view the resulting annotations in the source listing file to see what optimizations the compiler performed or else why the compiler was not able to optimize a particular code sequence. The default is /noannotations, which places no annotations in the source listing file. The /annotations option keywords are as follows:
Page 13 of 130
/annotations:none Same as /noannotations, the default value. /annotations:all Specifying /annotations all selects all of the following annotations. /annotations:code Specifying /annotations:code annotates the machine code listing with descriptions of special instructions used for prefetching, alignment, and so on. /annotations:detail Specifying /annotations:detail provides, where available, an additional level of annotation detail. /annotations:feedback Specifying /annotations:feedback provides information about any feedback optimizations (not available for Windows systems). /annotations:inlining Specifying /annotations:inlining indicates where code for a called procedure was expanded inline. /annotations:loop_transforms Specifying /annotations:loop_transforms indicates where advanced loop nest optimizations have been applied to improve cache performance (unroll and jam, loop fusion, loop interchange, and so on). /annotations:loop_unrolling Specifying /annotations:loop_unrolling indicates where a loop was unrolled (contents expanded multiple times). /annotations:prefetching Specifying /annotations:prefetching indicates where special instructions were used to reduce memory latency. /annotations:shrinkwrapping Specifying /annotations:shrinkwrapping indicates removal of code establishing routine context when it is not needed. /annotations:software_pipelining Specifying /annotations:software_pipelining indicates where instructions have been rearranged to make optimal use of the processor's functional units (not available for ia32 systems). /annotations:tail_calls
Page 14 of 130
Specifying /annotations:tail_calls indicates an optimization where a call from one routine to another can be replaced with a jump. /annotations:tail_recursion Specifying /annotations:tail_recursion indicates an optimization that eliminates unnecessary routine context for a recursive call.
The /annotations option is ignored unless you also specify the /list[:file] option or select the equivalent Source Listing option in the Listing File category in the visual development environment.
/architecture
Syntax: /architecture:keyword The /architecture (/arch) option controls the types of processor-specific instructions generated for this program unit. The /arch:keyword option uses the same keywords as the /tune:keyword option. All processors of a certain architecture type (ia32) implement a core set of instructions. Certain (more recent) processor versions include additional instruction extensions. Whereas the /tune:keyword option is primarily used by certain higher-level optimizations for instruction scheduling purposes, the /arch:keyword option determines the type of machine-code instructions generated for the program unit being compiled. In the visual development environment, specify the Generate Code For in the Code Generation Compiler Option Category. For ia32 (Intel and AMD) 32-bit processor systems, the supported /arch keywords are:
l
/arch:generic Generates code (sometimes called blended code) that is appropriate for processor generations for the architecture type in use. This is the default. Programs compiled on an ia32 system with the generic keyword will run on all ia32 systems.
/arch:host Generates code for the processor generation in use on the system being
Page 15 of 130
used for compilation. Depending on the host system used on ia32 systems, the program may or may not run on other ia32 systems. Using /arch:host on a: Intel Pentium processor system selects the pn1 keyword Intel Pentium Pro, Intel Pentium II, or AMD K6 processor system selects the pn2 keyword Intel Pentium III processor system selects the pn3 keyword AMD K6_2 or AMD K6_III processor system selects the k6_2 keyword AMD Athlon processor system selects the k7 keyword Intel Pentium 4 processor system selects the pn4 keyword
l
/arch:pn1 Generates code for the Pentium processor systems. Programs compiled with the pn1 keyword will run correctly on Pentium, Pentium Pro, Pentium II, Pentium III, AMD K6, and higher processors, but should not be run on 486 processors. The pn1 keyword replaces the p5 keyword (specifying /arch:pn1 and /arch:p5 are equivalent).
/arch:pn2 Generates code for the Pentium Pro, Pentium II, and AMD K6 processor systems only. Programs compiled with the pn2 or k6 keyword will run correctly on Pentium Pro, Pentium II, AMD K6, Pentium III, and higher processors, but should not be run on 486 or Pentium processors. The pn2 keyword replaces the p6 keyword (specifying /arch:pn2 and /arch:p6 are equivalent).
/arch:k6 Generates code for the AMD K6 (same as Pentium II systems) processor systems only. Programs compiled with the k6 or pn2 keyword will run correctly on Pentium Pro, Pentium II, AMD K6, Pentium III, and higher processors, but should not be run on 486 or Pentium processors.
/arch:pn3 Generates code for the Pentium III, AMD K6_2, and AMD K6_III processor systems only. Programs compiled with the pn3 keyword will run correctly on Pentium III, AMD K6_2, AMD K6_III, Pentium 4, and higher processors, but should not be run on 486, Pentium, Pentium Pro, Pentium II, or AMD K6 processors. The pn3 keyword replaces the p6p keyword (specifying /arch:pn3 and /arch:p6p are equivalent).
/arch:k6_2 Generates code for the AMD K6_2 and AMD K6_III processor systems.
Page 16 of 130
Programs compiled with the k6_2 keyword will run correctly on AMD K6_2, AMD K6_III, and AMD AthlonTM processors, but should not be run on 486, Pentium, Pentium Pro, Pentium II (same as AMD K6), Pentium III, or Pentium 4 processors.
l
/arch:k7 Generates code for the AMD Athlon processor systems only. Programs compiled with the k7 keyword will run correctly on AMD Athlon processors, but should not be run on 486, Pentium, Pentium Pro, Pentium II (same as AMD K6), Pentium III, Pentium 4, AMD K6_2, or AMD K6_III processors.
/arch:pn4 Generates code for the Pentium 4 processor systems only. Programs compiled with the pn4 keyword will run correctly on Pentium 4 processors, but should not be run on 486, Pentium, Pentium Pro, or Pentium II (same as AMD K6), Pentium III, AMD K6_2, AMD K6_III, or AMD Athlon processors.
Other processors (not listed) that have instruction-level compatibility with the processors listed above will have results similar to those processors. For ia64 (64-bit) systems, specify either /arch:generic or /arch:host. The following table shows the ia32 /arch:keywords and the systems that programs using these keywords can be used on:
keyword
AMD K6
AMD K6_2
AMD K6_III
AMD Athlon
Intel 486
Intel Pentium
Intel Pentium 4
* * *
* * * *
* * * *
* * * *
* X
* * X
* * * X
* * * *
Compiler and Linker Options pn4 k6 k6_2 k7 Legend: X * X * X * * X * indicates supported combinations of keyword and processor type * *
Page 17 of 130 X *
X indicates the best code generation combination of keyword and processor type
About timing program execution, see Analyze Program Performance. About the types of instructions used by the various keywords, see Requesting Code Generation for a Specific Processor Generation
/[no]asmattributes
Syntax: /asmattributes:keyword, /noasmattributes, /FA, /FAs, /FAc, or /FAcs The /asmattributes option indicates what information, in addition to the assembly code, should be generated in the assembly listing file. In the visual development environment, specify Assembly Options in the Listing File Compiler Option Category. The /asmattributes options are:
l
/asmattributes:source or /FAs Intersperses the source code as comments in the assembly listing file.
/asmattributes:machine or /FAc Lists the hex machine instructions at the beginning of each line of assembly code.
/asmattributes:all or /FAcs
Page 18 of 130
Intersperses both the source code as comments and lists the hex machine instructions at the beginning of each line of assembly code. This is equivalent to /asmattributes.
l
/asmattributes:none or /FA Provides neither interspersed source code comments nor a listing of hex machine instructions. This is equivalent to /noasmattributes.
If you omit the /asmattributes option, /asmattributes:none is used (default). The /asmattributes option is ignored if the /[no]asmfile[:file] option is not specified. The /FA, /FAs, /FAc, or /FAcs options can be used without the /[no] asmfile[:file] option.
/[no]asmfile
Syntax: /asmfile[:file], /noasmfile, /Fa[file], /Fc[file], /Fl[file], or /Fs[file] The /asmfile option or equivalent /Fx option indicates that an assembly listing file should be generated. If the file is not specified, the default filename used will be the name of the source file with an extension of .asm. In the visual development environment, specify Assembly Listing in the Listing File Compiler Option Category. When the /asmfile option or equivalent /Fx[file] option is specified and there are multiple source files being compiled, each source file will be compiled separately. Compiling source files separately turns off interprocedural optimization from being performed. When you specify /noasmfile or the /asmfile option is not specified, the compiler does not generate any assembly files. To specify the content of the assembly listing file, also specify /[no] asmattributes:keyword or specify the /Fx[file] options:
l
/FA[file] provides neither interspersed source code comments nor a listing of hex machine instructions. /FAs[file] provides interspersed source code as comments in the assembly listing file. /FAc[file] provides a list of hex machine instructions at the beginning of each line of assembly code. /FAcs[file] provides interspersed source code as comments and lists hex
Page 19 of 130
/assume
Syntax: /assume:keyword The /assume option specifies assumptions made by the Fortran syntax analyzer, optimizer, and code generator. These option keywords are: /assume:[no]accuracy_sensitive /assume:[no]buffered_io /assume:[no]byterecl /assume:[no]minus0 /assume:[no]source_include The /assume options are:
l
/assume:[no]accuracy_sensitive Specifying /assume:noaccuracy_sensitive allows the compiler to reorder code based on algebraic identities (inverses, associativity, and distribution) to improve performance. In the visual development environment, specify Allow Reordering of Floating-Point Operations in the Optimizations Compiler Option Category. The numeric results can be slightly different from the default (/assume:accuracy_sensitive) because of the way intermediate results are rounded. Numeric results with /assume:noaccuracy_sensitive are not categorically less accurate. They can produce more accurate results for certain floatingpoint calculations, such as dot product summations. For example, the following expressions are mathematically equivalent but may not compute the same value using finite precision arithmetic:
X = (A + B) - C X = A + (B - C)
If you omit /assume:noaccuracy_sensitive and omit /fast, the compiler uses a limited number of rules for calculations, which might prevent some optimizations. If you specify /assume:noaccuracy_sensitive, or if you specify /fast and
Page 20 of 130
omit /assume:accuracy_sensitive, the compiler can reorder code based on algebraic identities to improve performance. For more information on /assume:noaccuracy_sensitive, see Arithmetic Reordering Optimizations.
l
/assume:[no]buffered_io The /assume:buffered_io option controls whether records are written (flushed) to disk as each record is written (default) or accumulated in the buffer. For disk devices, /assume:buffered_io (or the equivalent OPEN statement BUFFERED='YES' specifier or the FORT_BUFFERED run-time environment variable) requests that the internal buffer will be filled, possibly by many record output statements (WRITE), before it is written to disk by the Fortran run-time system. If a file is opened for direct access, I/O buffering will be ignored. Using buffered writes usually makes disk I/O more efficient by writing larger blocks of data to the disk less often. However, if you request buffered writes, records not yet written to disk may be lost in the event of a system failure. Unless you set the FORT_BUFFERED environment variable to TRUE, the default is BUFFERED='NO' and /assume:nobuffered_io for all I/O, in which case, the Fortran run-time system empties its internal buffer for each WRITE (or similar record output statement). The OPEN statement BUFFERED specifier applies to a specific logical unit. In contrast, the /assume:[no]buffered_io option and the FORT_BUFFERED environment variable apply to all Fortran units. In the visual development environment, to enable /assume:buffered_io, specify the Enable I/O Buffering in the Optimizations Compiler Option Category. For more information on /assume:buffered_io, see Efficient Use of Record Buffers and Disk I/O.
/assume:[no]byterecl The /assume:byterecl option applies only to unformatted files. In the visual development environment, specify the Use Bytes as Unit for Unformatted Files in the Fortran Data Compiler Option Category. Specifying the /assume:byterecl option:
Page 21 of 130
Indicates that the units for an explicit OPEN statement RECL specifier value are in bytes. Forces the record length value returned by an INQUIRE by output list to be in byte units.
Specifying /assume:nobyterecl indicates that the units for RECL values with unformatted files are in four-byte (longword) units. This is the default.
l
/assume:[no]dummy_aliases Specifying the /assume:dummy_aliases option requires that the compiler assume that dummy (formal) arguments to procedures share memory locations with other dummy arguments or with variables shared through use association, host association, or common block use. The default is /assume:nodummy_aliases. In the visual development environment, specify Enable Dummy Argument Aliasing in the Fortran Data (or Optimizations) Compiler Option Category. These program semantics do not strictly obey the Fortran Standard and they slow performance. If you omit /assume:dummy_aliases, the compiler does not need to make these assumptions, which results in better run-time performance. However, omitting /assume:dummy_aliases can cause some programs that depend on such aliases to fail or produce wrong answers. You only need to compile the called subprogram with /assume:dummy_aliases. If you compile a program that uses dummy aliasing with /assume:nodummy_aliases in effect, the run-time behavior of the program will be unpredictable. In such programs, the results will depend on the exact optimizations that are performed. In some cases, normal results will occur; however, in other cases, results will differ because the values used in computations involving the offending aliases will differ. For more information, see Dummy Aliasing Assumption.
/assume:[no]minus0 This option controls whether the compiler uses Fortran 95 standard semantics for the IEEE floating-point value of -0.0 (minus zero) in the SIGN intrinsic, if the processor is capable of distinguishing the difference between -0.0 and +0.0. The default is /assume:nominus0, where the value -0.0 or +0.0 in the SIGN function is treated as 0.0. To request Fortran 95 semantics to allow use of the IEEE value -0.0 in the
Page 22 of 130
In the visual development environment, specify Enable IEEE Minus Zero Support in the Floating Point Compiler Option Category.
l
/assume:[no]protect_constants This option specifies whether constant actual arguments can be changed. By default, actual arguments that are constants are read-only (/assume:protect_constants). To allow changes to actual arguments that are constants, specify /assume:noprotect_constants. In the visual development environment, specify Constant Actual Arguments are Read-Only in the Fortran Data Compiler Option Category.
/assume:[no]source_include This option controls the directory searched for module files specified by a USE statement or source files specified by an INCLUDE statement:
Specifying /assume:source_include requests a search for module or include files in the directory where the source file being compiled resides. This is the default. Specifying /assume:nosource_include requests a search for module or include files in the current (default) directory.
In the visual development environment, specify the Default INCLUDE and USE Paths in the Preprocessor Compiler Option Category.
l
/assume:[no]underscore Specifying /assume:underscore option controls the appending of an underscore character to external user-defined names: the main program name, named COMMON, BLOCK DATA, and names implicitly or explicitly declared EXTERNAL. The name of blank COMMON remains _BLNK__, and Fortran intrinsic names are not affected. In the visual development environment, specify Append Underscore to External Names in the External Procedures (or Fortran Data) Compiler Option Category. Specifying /assume:nounderscore option does not append an underscore character to external user-defined names. This is the default.
For example, the following command requests the noaccuracy_sensitive and nosource_include keywords and accepts the defaults for the other /assume keywords:
Page 23 of 130
df /assume:(noaccuracy_sensitive,nosource_include)
testfile.f90
/[no]automatic
Syntax: /automatic, /noautomatic, /4Ya, or /4Na The /automatic or /4Ya option requests that local variables be put on the runtime stack. In the visual development environment, specify Variables Default to Automatic in the Fortran Data (or Optimizations) Compiler Option Category. The /noautomatic or /4Na option is the same as the /static option. The default is /noautomatic or /4Na, which causes all local variables to be statically allocated. If you specify /recursive, the /automatic (/4Ya) option is set.
/bintext
Syntax: /bintext:string or /Vstring Specifying /bintext (or /V) places the text string specified into the object file (.OBJ) being generated by the compiler. This string also gets propagated into the executable file. For example, the string might contain version number or copyright information. In the visual development environment, specify Object Text in the Code Generation Compiler Option Category. If the string contains a space or tab, the string must be enclosed by double quotation marks ("). A backslash (\) must precede any double quotation marks contained within the string. If the command line contains multiple /bintext or /V options, the last (rightmost) one is used. You can specify /nobintext to override previous /bintext or /V options on the same command line.
/[no]browser
Syntax: /browser[:filename], /nobrowser, or /FR
Page 24 of 130
The /browser or /FR option controls the generation of source browser information. When the /browser option is not specified, the compiler will not generate browser files (same as /nobrowser). In the visual development environment, specify Generate Source Browse Information in the General Compiler Option Category. Also, in the BrowseInfo tab, set Build Browse info check box. Browser information includes:
l l l l
Information about all the symbols in the source file The source code line in which a symbol is defined Each source code line where there is a reference to a symbol The relationships between calling functions and called functions
The default extension for source browser files is .SBR. The browser output is intended to be used as input to the Browse Information File Maintenance Utility (BSCMAKE), which generates a browse information file (.BSC) that can be examined in browse windows in the Microsoft visual development environment. Instead of using BCSMAKE, you can use the Project Settings dialog box in the visual development environment:
l l
Click the BrowseInfo tab. Set the Build browse info file check box.
When the /browser or /FR option is specified and there are multiple source files being compiled, each source file will be compiled separately. Compiling source files separately turns off interprocedure optimizations.
/ccdefault
Syntax: /ccdefault:keyword... The /ccdefault option specifies the default carriage control when a file is displayed on a terminal screen. In the visual development environment, specify the Default Output Carriage Control in the Run-Time Compiler Option Category. The /ccdefault keywords are as follows:
l
/ccdefault:default
Page 25 of 130
Allows other options, such as Compatibility option /vms, to affect this default setting: If /vms is specified with /ccdefault:default, carriage control defaults to FORTRAN (/ccdefault:fortran) if the file is formatted, and the unit is connected to a terminal. If /novms (default) is specified with /ccdefault:default, carriage control defaults to LIST (/ccdefault:list). /ccdefault:fortran
Uses normal Fortran interpretation of the first character, such as the character "0" resulting in a blank line before output.
l
/[no]check
Syntax: /check:keyword, /nocheck, /4Yb, /4Nb The /check, /4Yb, or /4Nb options control whether extra code is generated for certain run-time checking. Run-time checks can result in issuing run-time messages for certain conditions. In the visual development environment, specify the Runtime Error Checking items in the Run time Compiler Option Category. The /check options are as follows: /check:arg_temp_created (Actual Arguments use Temporary Storage) /check:flawed_pentium (ia32 systems) (Flawed Pentium) /check:output_conversion (Edit Descriptor Data Size) /check:bounds (Array and String bounds) /check:format (Edit Descriptor Data Type) /check:overflow (Integer Overflow)
Compiler and Linker Options /check:power (Power Operations) /4Yb /check or /check:all
If you omit these options, the default is: /check: (noarg_temp_created,nobounds,flawed_pentium,noformat,nopower,nooutput_conversion,n When using the visual development environment debug configuration, the default for bounds checking changes from /check:nobounds to /check:bounds. The /check keywords and /4Yb, and /4Nb options are:
l
/check:arg_temp_created Requests that a run-time informational message appear if actual arguments are copied into temporary storage before routine calls. Specifying /check:noarg_temp_created (the default) does not generate the extra code needed to display a message run-time when this condition occurs.
/check:bounds Requests a run-time error message if a reference to an array subscript or character substring is outside of the declared bounds. The default for the command line and the release configuration (visual development environment) is /check:nobounds, which does not issue a run-time message for this condition. The default for the debug configuration is /check:bounds, which issues a run-time message for an out-of-bounds array subscript or character substring.
/check:flawed_pentium (ia32 systems) On ia32 systems, requests a run-time error message if a flawed Pentium (586) processor is detected. The default is /check:flawed_pentium, which does issue a run-time error message for this condition and stops program execution. To allow program execution to continue when this condition occurs, set the environment variable FOR_RUN_FLAWED_PENTIUM to true and rerun the program (see Run-Time Environment Variables). For more information on the Pentium flaw, see Intel Pentium Floating-Point Flaw. You can also use the FOR_CHECK_FLAWED_PENTIUM routine.
Page 27 of 130
/check:format Requests a run-time error message when the data type for an item being formatted for output does not match the FORMAT descriptor. Specifying /check:noformat suppresses the run-time error message for this condition.
/check:output_conversion Requests a run-time message (number 63) when format truncation occurs (when a number is too large to fit in the specified format field length without loss of significant digits). Specifying /check:nooutput_conversion does not display the message when format truncation occurs.
/check:overflow Requests a continuable run-time message when integer overflow occurs. Specifying /check:nooverflow suppresses the run-time message.
/check:power Requests a run-time error message for arithmetic expressions 0.0 ** 0.0 and negative-value ** integer-value-of-type-real. The default is /check:nopower (suppress the run-time error message), allowing the calculations to occur, where 0.0 ** 0.0 equals 1.0 and (-3.0) ** 3.0 equals -27.0. You can specify /check:power to allow a run-time error message to be issued for this type of expression.
/check:underflow Requests an informational run-time message when floating-point underflow occurs when /fpe:0 is specified (/check:underflow is not supported with /fpe:3). Specifying /check:nounderflow suppresses a run-time message when floating-point underflow occurs.
Page 28 of 130
/check or /check:all
/[no]comments
Syntax: /comments or /nocomments The /comments option writes C-style comments to the output file. The /nocomments option does not write C-style comments to the output file. This option applies only to the FPP preprocessor. For more information, type FPP /? to view FPP options.
/[no]compile_only
Syntax: /compile_only, /nocompile_only, or /c The /compile_only or /c option suppresses linking. The default is /nocompile_only (perform linking). If you specify the /compile_only option at higher levels of optimization and also specify /object:filename, the /object:filename option causes multiple Fortran input files (if specified) to be compiled into a single object file. This allows interprocedural optimizations to occur. However, if you use multiple source files and the /compile_only option without the /object:file option, multiple object files are created and interprocedural optimizations do not occur. In the visual development environment, to compile (not link) a source file: 1. In the FileView pane, select (highlight) the file to be compiled 2. From the Build menu, select Compile filename.xxx
/convert
Syntax:
Page 29 of 130
/convert:keyword The /convert option specifies the format of unformatted files containing numeric data. On ia32 and ia64 systems, the format used in memory is always IEEE little endian format. If you want to read and write unformatted data in IEEE little endian format, you do not need to convert your unformatted data and can omit this option (or specify /convert:native). This method affects all unit numbers that use unformatted data specified by the program. Other methods are available to specify the format for all or certain unformatted files (see Converting Unformatted Numeric Data). In the visual development environment, specify the Unformatted File Conversion in the Compatibility Compiler Option Category. The /convert options are: /convert:big_endian /convert:cray /convert:ibm /convert:native /convert:vaxg
l
/convert:little_endian /convert:vaxd
/convert:big_endian Specifies that unformatted files containing numeric data are in IEEE big endian (nonnative) format. The resulting program will read and write unformatted files containing numeric data assuming the following:
Big endian integer format (INTEGER declarations of the appropriate size). Big endian IEEE floating-point formats (REAL and COMPLEX declarations of the appropriate size).
/convert:cray Specifies that unformatted files containing numeric data are in CRAY (nonnative) big endian format. The resulting program will read and write unformatted files containing numeric data assuming the following:
Big endian integer format (INTEGER declarations of the appropriate size). Big endian CRAY proprietary floating-point formats (REAL and COMPLEX declarations of the appropriate size).
/convert:ibm
Page 30 of 130
Specifies that unformatted files containing numeric data are in IBM (nonnative) big endian format. The resulting program will read and write unformatted files containing numeric data assuming the following:
Big endian integer format (INTEGER declarations of the appropriate size). Big endian IBM proprietary floating-point formats (REAL and COMPLEX declarations of the appropriate size).
/convert:little_endian Specifies that numeric data in unformatted files is in native little endian integer format and IEEE little endian floating-point format (same as used in memory), as follows:
Integer data is in native little endian format. REAL(KIND=4) and COMPLEX(KIND=4) (SINGLE PRECISION) data is in IEEE little endian S_floating format. REAL(KIND=8) and COMPLEX (KIND=8) (DOUBLE PRECISION) data is in IEEE little endian T_floating format.
/convert:native Specifies that numeric data in unformatted files is not converted. This is the default.
/convert:vaxd Specifies that numeric data in unformatted files is in VAXD little endian format, as follows:
Integer data is in native little endian format. REAL(KIND=4) and COMPLEX(KIND=4) (SINGLE PRECISION) data is in VAX F_floating format. REAL(KIND=8) and COMPLEX (KIND=8) (DOUBLE PRECISION) data is in VAX D_floating format.
/convert:vaxg Specifies that numeric data in unformatted files is in VAXG little endian format, as follows:
Integer data is in native little endian format. REAL(KIND=4) and COMPLEX(KIND=4) (SINGLE PRECISION) data is in VAX F_floating format. REAL(KIND=8) and COMPLEX(KIND=8) (DOUBLE PRECISION) data is
Page 31 of 130
For more information on unformatted file conversion, see Converting Unformatted Numeric Data.
/cxml
Syntax: /cxml The /cxml option requests that the Compaq Extended Math Library (CXML) library be passed to the linker. Your program still needs to specify the INCLUDE 'CXML_INCLUDE.F90' statement. The /cxml option is provided for documentation purposes only. For more information, see Using the Compaq Extended Math Library (CXML). In the visual development environment, specify the Use Compaq Extended Math Library (CXML) in the Library Compiler Option Category.
/[no]d_lines
Syntax: /d_lines, /nod_lines, /4ccD, or /4ccd The /d_lines, /4ccD, or /4ccd options indicate that lines in fixed-format files that contain a D in column 1 should be treated as source code. Specifying /nod_lines (the default) indicates that these lines are to be treated as comment lines. In the visual development environment, specify Compile DEBUG (D) Lines in the Debug Compiler Option Category. The compiler does not support the use of characters other than a D or d with the /4ccstring (see the dlines Sample program in ...\Samples\Advanced\com\).
/[no]dbglibs
Syntax: /dbglibs or /nodbglibs The /dbglibs option controls whether the debug version or the non-debug version of the C run-time library is linked against. The default is /nodbglibs, which will link against the non-debug version of the C library, even
Page 32 of 130
If you specify /debug:full for an application that calls C library routines and you need to debug calls into the C library, you should also specify /dbglibs to request that the debug version of the library be linked against. In the visual development environment, specify one of the Debug library options in the Use Fortran Run-Time Libraries in the Libraries Compiler Option Category. When you specify /dbglibs, the C debug library linked against depends on the specified /libs:keyword and /[no]threads options, and is one of: libcd.lib, libcmtd.lib, or msvcrtd.lib. For More Information:
l l
/[no]debug
Syntax: /debug:keyword, /nodebug, /Z7, /Zd, or /Zi The /debug, /Z7, /Zd, or /Zi options control the level of debugging information associated with the program being compiled. In the visual development environment, specify the Debugging Level in the General or Debug Compiler Option Category. The options are: /debug:none or /nodebug /debug:minimal or /Zd /debug:partial The /debug options:
l
/debug:none or /nodebug If you specify /debug:none or /nodebug, the compiler produces no symbol table information, which is needed for debugging or profiling. Only symbol information needed for linking (global symbols) is produced. The size of the resulting object module is the minimum size. If this option is specified, /debug:none is passed to the linker.
/debug:minimal or /Zd
Page 33 of 130
If you specify /debug:minimal or /Zd, the compiler produces minimal debug information, which allows global symbol table information needed for linking, but not local symbol table information needed for debugging. If /debug:minimal is specified, /debug:minimal and /debugtype:cv is passed to the linker. If you omit the /[no]debug:keyword, /Z7, /Zd, and /Zi options, /debug:minimal is the default on the command line and for a release configuration in the visual development environment. The /Zd option implies /nopdbfile and passes /debug:minimal /pdb:none /debugtype:cv to the linker. The object module size is somewhat larger than if you specified /debug:none, but is smaller than if you specified /debug:full.
l
/debug:partial If you specify /debug:partial, the compiler produces debugging information to allow global symbol table information needed for linking, but not local symbol table information needed for debugging. If /debug:partial is specified, /debug:partial /debugtype:cv /pdb:none is passed to the linker. The object module size is somewhat larger than if you specified /debug:none, but is smaller than if you specified /debug:full.
/debug:full, /debug, /Zi, or /Z7 If you specify /debug:full, /debug, /Zi, or /Z7, the compiler produces symbol table information needed for full symbolic debugging of unoptimized code and global symbol information needed for linking. This is the default for a debug configuration in the visual development environment. If you specify /debug:full for an application that make calls to C library routines and you need to debug calls into the C library, you should also specify /dbglibs to request that the appropriate C debug library be linked against (see Specifying Consistent Library Types). The /Z7 option implies /nopdbfile and passes /debug:full /debugtype:cv /pdb:none to the linker. The /debug:full, /debug, and /Zi options imply /pdbfile and pass /debug:full and /debugtype:cv to the linker. If you specify /debug (with no keyword), the default optimization level
Page 34 of 130
changes to /optimize:0 (instead of /optimize:4) for the DF command. To request program counter run-time correlation to source file line numbers (full traceback) for severe run-time errors, specify the /traceback option.
/define
Syntax: /define:symbol[=value] The /define option defines the symbol specified for use with conditional compilation directives or the Fortran preprocessor, FPP. A value specified for /define can be a character or integer value. If a value is not specified, 1 is assigned to symbol. When only using the Fortran preprocessor FPP, to request that symbol values defined by /define apply only to FPP and are not seen by compiler directives, also specify /nodefine on the DF command line. In the visual development environment, specify the Predefined Preprocessor Symbols in the General or Preprocessor Compiler Option Category. You can use the directives to detect symbol definitions, such as the IF Directive Construct. Like certain other compiler options, an equivalent directive exists (DEFINE directive). The following preprocessor symbols are predefined by the compiler system and are available to compiler directives and FPP (except _DF_VERSION_ and _VF_VERSION_): Predefined Preprocessor Symbols Predefined Symbol Name and Value _DF_VERSION_=660 (660 for Version 6.6) _DLL=1 Conditions When this Symbol is Defined Compiler only Only if /libs:dll, /MDs, /MD, /dll, or /LD is specified, but not when /libs:static is specified Only for ia32 (32-bit) systems Only for ia64 (64-bit) systems
_INTEGRAL_MAX_BITS=32 _INTEGRAL_MAX_BITS=64
Compiler and Linker Options _ITANIUM_A3_=1 _MSFORTRAN_=401 _MT=1 _M_IX86=500 _M_IA64=64100 _VF_VERSION_=660 (660 for Version 6.6) _WIN32=1 _WIN64=1 _WIN95=1 _WIN98=1 _WINME=1 _X86_=1 Only for ia64 (64-bit) systems
Page 35 of 130
Only if /fpscomp:symbols is specified or you use the FL32 command Only if /threads or /MT is specified Only for ia32 (32-bit) systems Only for ia64 (64-bit) systems Compiler only Always defined (both ia32 and ia64 systems) Only for ia64 (64-bit) systems Only for ia32 (32-bit) systems running the Windows 95 operating system Only for ia32 (32-bit) systems running the Windows 98 operating system Only for ia32 (32-bit) systems running the Windows Me operating system Only for ia32 (32-bit) systems
When using the non-native ia64 compiler, platform-specific symbols are set for the target platform of the executable, not for the system in use.
/dll
Syntax: /dll[:file], /nodll, or /LD The /dll or /LD option indicates that the program should be linked as a DLL file. The /dll or /LD option overrides any specification of the run-time routines to be used and activates the /libs:dll option. A warning is generated when the /libs=qwin or /libs=qwins option and /dll option are used together. In the visual development environment, specify the project type as Fortran Dynamic Link Library (DLL). If you omit file, the /dll or /LD option interacts with the /exe and the /Fe
Page 36 of 130
If neither /exe nor /Fe is specified, the first file name used on the command line is used with an extension of .DLL. If either /exe:file or /Fefile is specified with a file name, that name is used for the DLL file. If the specified file name does not end with a "." or have an extension, an extension of .DLL is added to it.
To request linking with multithreaded libraries, specify the /threads option. For information about building DLL files from the visual development environment, see Fortran Dynamic-Link Library Projects and Creating Fortran DLLs. For a list of Fortran PowerStation style options (such as /LD and /MDs) and their DF command equivalents, see Equivalent Visual Fortran Compiler Options.
/[no]error_limit
Syntax: /error_limit[:count] or /noerror_limit The /error_limit option specifies the maximum number of error-level or fatallevel compiler errors allowed for a given file before compilation aborts. If you specify /noerror_limit (command line), there is no limit on the number of errors that are allowed. In the visual development environment, specify the Compilation Error Limit in the Compilation Diagnostics Compiler Option Category. The default is /error_limit:30 or a maximum of 30 error-level and fatal-level messages. If the maximum number of errors is reached, a warning message is issued and the next file (if any) on the command line is compiled.
/[no]exe
Syntax: /exe[:file], /noexe, or /Fefile The /exe or /Fe option specifies the name of the executable program (EXE) or dynamic-link library (DLL) file being created. To request that a DLL be created instead of an executable program, specify the /dll option.
/[no]extend_source
Page 37 of 130
Syntax: /extend_source[:size], /noextend_source, or /4Lsize The /extend_source or /4Lsize option controls the column used to end the statement field in fixed-format source files. When a size is specified, that will be the last column parsed as part of the statement field. Any columns after that will be treated as comments. Specifying /extend_source (or /4L132 or /4L80) sets the /fixed option. In the visual development environment, specify the Fixed-Form Line Length in the Fortran Language Compiler Option Category. The following options are equivalent:
l
l l
/noextend_source, /extend_source:72, or /4L72 specify the last column as 72. /extend_source:80 or /4L80 specify the last column as 80. /extend_source, /extend_source:132, or /4L132 specify the last column as 132.
/extfor
Syntax: /extfor:ext The /extfor option specifies file extensions (ext) to be processed by the Compaq Fortran compiler. One or more file extensions can be specified. A leading period before each extension is optional (for and .for are equivalent).
/extfpp
Syntax: /extfpp:ext The /extfpp option specifies file extensions (ext) to be processed by the FPP preprocessor. One or more file extensions can be specified. A leading period before each extension is optional (fpp and .fpp are equivalent).
/extlnk
Syntax:
Page 38 of 130
/extlnk:ext The /extlnk option specifies file extensions (ext) to be processed by the linker. One or more file extensions can be specified. A leading period before each extension is optional (obj and .obj are equivalent).
/[no]f66
Syntax: /f66 or /nof66 The /f66 option requests that the compiler select FORTRAN-66 interpretations in cases of incompatibility (default is /nof66). Differences include the following:
l
DO loops are always executed at least once (see Execution of DO Constructs) FORTRAN-66 EXTERNAL statement syntax and semantics are allowed (see FORTRAN-66 Interpretation of the External Statement) If the OPEN statement STATUS specifier is omitted, the default changes to STATUS='NEW' instead of STATUS='UNKNOWN' If the OPEN statement BLANK specifier is omitted, the default changes to BLANK='ZERO' instead of BLANK='NULL'
In the visual development environment, specify Enable FORTRAN-66 Semantics in the Fortran Language Compiler Option Category.
/[no]f77rtl
Syntax: /f77rtl or /nof77rtl The /f77rtl option controls the run-time support that is used when a program is executed. Specifying /f77rtl uses the Compaq Fortran 77 run-time behavior. In the visual development environment, specify Enable F77 Run-Time Compatibility in the Compatibility Compiler Option Category. Specifying /nof77rtl uses the Visual Fortran (Compaq Fortran) run-time behavior. Unless you specify /f77rtl, /nof77rtl is used.
/fast
Syntax:
Page 39 of 130
/fast The /fast option sets several options that generate optimized code for fast runtime performance. Specifying this option is equivalent to specifying:
l l l l l
If any of the /stand options are also specified, /align:dcommons and /align:sequence are not set. In the visual development environment, specify the Generate Most Optimized Code in the Code Generation Compiler Option Category. If you omit /fast, these performance-related options will not be set.
/[no]fixed
Syntax: /fixed, /nofixed, /4Nf, or /4Yf The /fixed or /4Nf option specifies that the source file should be interpreted as being in fixed-source format. Equivalent options are as follows:
l
The /fixed, /nofree, and /4Nf options are equivalent and request fixedsource form. The /nofixed, /free, and /4Yf options are equivalent and request freesource form.
In the visual development environment, specify the Source Form in the Fortran Language Compiler Option Category. If you omit /[no]free, /[no]fixed, /4Nf, and /4Yf, the compiler assumes:
l l
Files with an extension of .f90, .F90, or .i90 are free-format source files. Files with an extension of .f, .for, .FOR, or .i are fixed-format files.
Page 40 of 130
The /fltconsistency or /Op option enables improved floating-point consistency on ia32 systems. Floating-point operations are not reordered and the result of each floating-point operation is stored into the target variable rather than being kept in the floating-point processor for use in a subsequent calculation. This option is ignored on ia64 systems. In the visual development environment, specify Enable Floating-Point Consistency in the Floating Point Compiler Option Category. The default is /nofltconsistency, which provides better accuracy and run-time performance at the expense of less consistent floating-point results.
/[no]fpconstant
Syntax: /fpconstant or /nofpconstant The /fpconstant option requests that a single-precision constant assigned to a double-precision variable be evaluated in double precision. If you omit /fpconstant (or specify the default /nofpconstant), a single-precision constant assigned to a double-precision variable is evaluated in single precision. The Fortran 90 standard requires that the constant be evaluated in single precision. In the visual development environment, specify Extended Precision of SinglePrecision Constants in the Floating Point (or Fortran Data) Compiler Option Category. Certain programs created for FORTRAN-77 compilers (including Compaq Fortran 77) may show different floating-point results, because they rely on singleprecision constants assigned to a double-precision variable to be evaluated in double precision. In the following example, if you specify /fpconstant, identical values are assigned to D1 and D2. If you omit the /fpconstant option, the compiler will obey the standard and assign a less precise value to D1:
REAL (KIND=8) D1, D2 DATA D1 /2.71828182846182/ DATA D2 /2.71828182846182D0/ ! REAL (KIND=4) value expanded to double ! Double value assigned to double
/fpe
Syntax:
Page 41 of 130
/fpe:level The /fpe:level option controls floating-point exception handling at run time for the main program. This includes whether exceptional floating-point values are allowed and how precisely run-time exceptions are reported. The /fpe:level option specifies how the compiler should handle the following floating-point exceptions:
l
l l
When floating-point calculations result in a divide by zero, overflow, or invalid operation. When floating-point calculations result in an underflow. When a denormalized number or other exceptional number (positive infinity, negative infinity, or a NaN) is present in an arithmetic expression
On ia32 systems, the default is /fpe:3. Using /fpe:0 will slow run-time performance on ia32 systems. On ia64 systems, the default is /fpe:0 (many programs do not need to handle denormalized numbers or other exceptional values). Using /fpe:3 will slow run-time performance on ia64 systems.
In the visual development environment, specify the Floating-Point Exception Handling in the Floating Point Compiler Option Category. The /fpe:level (level is 0 or 3) options are as follows: Handling of Divide by Zero, Overflow, and Invalid Operation Exceptional values are not allowed. The program terminates after displaying a message.
/fpe:level Option
Handling of Underflow
Sets any calculated denormalized value (result) to zero and lets the program continue. Any use of a denormalized number in an arithmetic expression uses a value of 0.0 and execution continues. A message is displayed only if /check:underflow is also specified.
Compiler and Linker Options /fpe:3 (default on ia32 systems) Leaves any calculated denormalized value as is. The program continues, allowing gradual underflow. Use of a denormalized (or exceptional) number in an arithmetic expression results in program continuation, but with slower performance.
Page 42 of 130 The program continues. No message is displayed. A NaN or Infinity (+ or -) will be generated.
When compiling different routines in a program separately, you should use the same /fpe:level value. On ia32 systems, the /fpe option, /check:underflow option, and MATHERRQQ routine interact as follows: Is a UserWritten MATHERRQQ Routine Present? No Underflow Handling by the Visual Fortran RunTime System on ia32 Systems The underflowed result is set to zero (0). The program continues. The underflowed result is set to zero (0). The program continues. The underflowed result is set to zero (0). The program continues. The number of underflowed results are counted and messages are displayed for the first two occurrences. The underflowed result is set to zero (0). The program continues. The number of underflowed results are counted and messages are displayed for the first two occurrences.
/fpe:0
No
Yes
/fpe:0
Yes
No
/fpe:0
Yes
Yes
Page 43 of 130 Denormalized results are allowed and the program continues. Traps are masked and no handlers are invoked. Denormalized results are allowed and the program continues. Traps are masked and no handlers are invoked. For Version 6, a fatal error results and the program terminates. Depends on the source causing the underflow:
l
/fpe:3
No
Yes
/fpe:3
Yes
No
/fpe:3
Yes
Yes
If the underflow occurs in an intrinsic procedure, the undefined result is left as is. The program continues with the assumption that the userspecified MATHERRQQ handler will perform any result fix up needed. If the underflow does not occur in an intrinsic procedure, for Version 6.0, a fatal error results and the program terminates.
About the /fpe option on ia32 systems, see How the Floating-Point Exception Handling (/fpe) Compiler Option Works. About the floating-point environment and the MATHERRQQ routine (ia32 systems), see The Floating-Point Environment.
Page 44 of 130
About creating exception and termination handlers, see Advanced Exception and Termination Handling Considerations. About routines that can obtain or set the floating-point exception settings used by Visual Fortran at run-time, see FOR_SET_FPE and FOR_GET_FPE. About IEEE floating-point exception handling, see the IEEE Standard for Binary Floating-Point Arithmetic (ANSI/IEEE Standard 754-1985).
/fpp
Syntax: /fpp[:"options"] The /fpp option activates the FPP preprocessor and optionally passes options to FPP as is. The FPP preprocessor can process both free- and fixed-form Fortran source files. Alternatively, you can use compiler directives, such as the IF Directive Construct, to detect symbol definitions and perform conditional compilation. You can run FPP:
l
On the DF command line, by adding the /fpp option. By default, the specified files are compiled and linked. To retain the intermediate (.i or .i90) file, specify the /keep option. In the visual development environment, by specifying the Use FPP option in the Preprocessor Compiler Option Category. By default, the file is compiled and linked. To retain the intermediate (.i or .i90) file, specify the /keep option on the command line or (in the visual development environment, Project Settings dialog box) the Project Options: box. On the command line, by using the FPP command. In this case, the compiler is not invoked. When using the FPP command line, you need to specify the input file and the output intermediate (.i or .i90) file.
FPP is a modified version of the ANSI C preprocessor and supports a similar set of directives (including syntax and semantics). It supports the following directives: #define, #elif, #else, #endif, #if, #ifdef, #ifndef, #include, and #undef. For example, the following DF command invokes FPP, specifies the /noC option to FPP, uses the /define option to define the symbol testcase, and preprocesses file cond.for before it is compiled and linked:
DF /fpp:"/noC" /define:testcase=2 cond.for
For a list of predefined preprocessor symbols (such as _X86_), see /define. For information on FPP options, type FPP /HELP on the command line.
Page 45 of 130
/[no]fpscomp
Syntax: /fpscomp[:keyword...] or /nofpscomp The /fpscomp option controls whether certain aspects of the run-time system and semantic language features within the compiler are compatible with Visual Fortran or Microsoft Fortran PowerStation. If you experience problems when porting applications from Fortran PowerStation, specify /fpscomp:keyword (or /fpscomp:all). When porting applications from Compaq Fortran, use /fpscomp:none or /fpscomp:libs (the default). In the visual development environment, specify the PowerStation 4.0 Compatibility Options in the Compatibility Compiler Option Category. The /fpscomp options and their visual development environment names are: /fpscomp:[no]filesfromcmd (Filenames from Command Line) /fpscomp:[no]ioformat (I/O Format) /fpscomp:[no]libs (Libraries) /fpscomp:[no]general (Other Run-time Behavior) /fpscomp:[no]ldio_spacing (List Directed I/O Spacing) /fpscomp:[no]logicals (Logical Values)
/fpscomp:[no]symbols /fpscomp:all and /fpscomp (Predefined Preprocessor Symbols) /nofpscomp or /fpscomp:none /fpscomp:[no]filesfromcmd Specifying /fpscomp:filesfromcmd for a file where the OPEN statement FILE specifier is blank (FILE=' '), requests that the following actions be taken at runtime:
l
The program reads a filename from the list of arguments (if any) in the command line that invoked the program. If any of the command-line arguments contain a null string (''), the program asks the user for the corresponding filename. Each additional OPEN statement with a nameless FILE specifier reads the next command-line argument. If there are more nameless OPEN statements than command-line arguments, the program prompts for additional file names.
Page 46 of 130
In a QuickWin application, a File Select dialog box appears to request file names.
Specifying /fpscomp:nofilesfromcmd disables the run-time system from using the filename specified on the command line when the OPEN statement FILE specifier is omitted, allowing the application of default directory, file name, and extensions like Compaq Fortran, such as the FORTn environment variable and the FORT.n file name (where n is the unit number). Specifying /fpscomp:filesfromcmd affects the following Fortran features:
l
The OPEN statement FILE specifier For example, assume a program OPENTEST contains the following statements:
OPEN(UNIT = 2, FILE = ' ') OPEN(UNIT = 3, FILE = ' ') OPEN(UNIT = 4, FILE = ' ')
The following command line assigns the file TEST.DAT to Unit 2, prompts the user for a filename to associate with Unit 3, then prompts again for a filename to associate with Unit 4:
opentest test.dat '' ''
Implicit file open statements such as the WRITE, READ, and ENDFILE statements Unopened files referred to in READ or WRITE statements are opened implicitly as if there had been an OPEN statement with a name specified as all blanks. The name is read from the command line.
WRITE(UNIT = 8, FMT='(2I5)') int1, int2 ! Where "8" has not been ! explicitly associated with a file.
For more information about running Visual Fortran programs with the /fpscomp:filesfromcmd option or the use of alternative routines, see Running Fortran Applications. /fpscomp:[no]general Specifying /fpscomp:[no]general controls which run-time behavior is used when a difference exists between Visual Fortran and Microsoft Fortran PowerStation and either semantic must remain available for compatibility reasons:
Page 47 of 130
Specify /fpscomp:general to request Microsoft Fortran PowerStation semantics. Specify /fpscomp:nogeneral to request Visual Fortran semantics.
The BACKSPACE statement: Allows files opened with ACCESS='APPEND' to be used with the BACKSPACE statement. Allows files opened with ACCESS='DIRECT' to be used with the BACKSPACE statement. Note: Allowing files that are not opened with sequential access (such as ACCESS='DIRECT') to be used with the BACKSPACE statement violates the Fortran 90 standard and may be removed in the future. Section 9.5 states the following: "A file that is not connected for sequential access must not be referred to by a BACKSPACE, an ENDFILE, or a REWIND statement..."
The READ statement: Formatted: READ(eunit, format [, advance][, iostat]...) Reading from a formatted file opened for direct access will read records that have the same record type format as Fortran PowerStation when /fpscomp:general is set. This consists of accounting for the trailing Carriage Return/Line Feed pair (<CR><LF>) that is part of the record. Allows sequential reads from a formatted file opened for direct access. Note: Allowing files that are not opened with sequential access (such as ACCESS='DIRECT') to be used with the sequential READ statement violates the Fortran 90 standard and may be removed in the future. Section 9.2.1.2.2 states the following: "Reading and writing records is accomplished only by direct access input/output statements."
Allows the last record in a file opened with FORM='FORMATTED' and a record type of STREAM_LF or STREAM_CR that does not end with a proper record terminator (<line feed> or <carriage return>) to be read without producing an error. Unformatted: READ(eunit [, iostat]...) Allows sequential reads from an unformatted file opened for direct access. Note: Allowing files that are not opened with sequential access (such as ACCESS='DIRECT') to be read with the sequential READ statement violates the Fortran 90 standard and may be removed in the future. Section 9.2.1.2.2 states the following: "Reading and writing records is
Page 48 of 130
The INQUIRE statement: The CARRIAGECONTROL specifier returns the value "UNDEFINED" instead of "UNKNOWN" when the carriage control is not known and when /fpscomp:general is set. The NAME specifier returns the file name "UNKNOWN" instead of space filling the file name when the file name is not known and when /fpscomp:general is set. The SEQUENTIAL specifier returns the value "YES" instead of "NO" for a direct access formatted file when /fpscomp:general is set. The UNFORMATTED specifier returns the value "NO" instead of "UNKNOWN" when it is not known whether unformatted I/O can be performed to the file and when /fpscomp:general is set. Note: Returning the value "NO" instead of "UNKNOWN" for this specifier violates the Fortran 90 standard and may be removed in the future. See Section 9.6.1.12.
The OPEN statement: If a file is opened with an unspecified STATUS keyword value, and is not named (no FILE specifier), the file is opened as a scratch file when /fpscomp:general is set. For example:
OPEN (UNIT = 4)
In contrast, when /fpscomp:nogeneral is in effect with an unspecified STATUS value with no FILE specifier, the FORTn environment variable and the FORT.n file name are used (where n is the unit number). If the STATUS value was not specified and if the name of the file is "USER", the file is marked for deletion when it is closed when /fpscomp:general is set. Allows a file to be opened with the APPEND and READONLY characteristics when /fpscomp:general is set. If the CARRIAGECONTROL specifier is defaulted, gives "LIST" carriage control to direct access formatted files instead of "NONE" when /fpscomp:general is set. Gives an opened file the additional default of write sharing when /fpscomp:general is set. Gives the file a default block size of 1024 when /fpscomp:general is set as compared to 8192 (see Efficient Use of Record Buffers and Disk I/O). If the MODE and ACTION specifier is defaulted and there was an error opening the file, then try opening the file read only, then write only. If the CARRIAGECONTROL specifier is defaulted and if the device type
Page 49 of 130
is a terminal file, the file is given the default carriage control value of "FORTRAN" as compared to "LIST" when /fpscomp:general is set. If a file that is being re-opened has a different file type than the current existing file, an error is returned when /fpscomp:general is set. Gives direct access formatted files the same record type as Fortran PowerStation when /fpscomp:general is set. This means accounting for the trailing Carriage Return/Line Feed pair (<CR><LF>) which is part of the record.
Writes the Fortran PowerStation output string and/or returns the same exit condition values when /fpscomp:general is set.
The WRITE statement: Formatted: WRITE(eunit, format [, advance][, iostat]...) n Writing to formatted direct files When writing to a formatted file opened for direct access, records are written in the same record type format as Fortran PowerStation when /fpscomp:general is set. This consists of adding the trailing Carriage Return/Line Feed pair (<CR><LF>) that is part of the record. Ignores the CARRIAGECONTROL specifier setting when writing to a formatted direct access file.
n
Interpreting Fortran carriage control characters When interpreting Fortran carriage control characters during formatted I/O, carriage control sequences are written which are the same as Fortran PowerStation when /fpscomp:general is set. This is true for the "Space, 0, 1 and + " characters.
Performing non-advancing I/O to the terminal. When performing non-advancing I/O to the terminal, output is written in the same format as Fortran PowerStation when /fpscomp:general is set.
Interpreting the backslash (\) and dollar ($) edit descriptors When interpreting backslash and dollar edit descriptors during formatted I/O, sequences are written the same as Fortran PowerStation when /fpscomp:general is set.
Page 50 of 130
Allows sequential writes from an unformatted file opened for direct access. Note: Allowing files that are not opened with sequential access (such as ACCESS='DIRECT') to be read with the sequential WRITE statement violates the Fortran 90 standard and may be removed in the future. Section 9.2.1.2.2 states the following: "Reading and writing records is accomplished only by direct access input/output statements."
l
/fpscomp:[no]ioformat Controls which run-time behavior is used for the semantic format for listdirected formatted I/O and unformatted I/O. Specify /fpscomp:ioformat to request Microsoft Fortran PowerStation semantic conventions and record formats (see Microsoft Fortran PowerStation Compatible Files). Specify /fpscomp:noioformat to request Compaq Fortran semantic conventions. This affects the following Fortran features:
l
The WRITE statement: Formatted List-Directed: WRITE(eunit, * [, iostat]...) Formatted Internal List-Directed: WRITE(iunit, * [, iostat]...) Formatted Namelist: WRITE(eunit, nml-group [, iostat]...) If /fpscomp:ioformat is set, the output line, field width values, and the list-directed data type semantics are dictated according to the following sample for real constants (N below):
n
For 1 <= N < 10**7, use F15.6 for single precision or F24.15 for double. For N < 1 or N >= 10**7, use E15.6E2 for single precision or E24.15E3 for double.
See the Fortran PowerStation documentation for more detailed information about the other data types affected.
Unformatted: WRITE(eunit [, iostat]...) If /fpscomp:ioformat is set, the unformatted file semantics are dictated according to the Fortran PowerStation documentation. Be aware that the file format differs from that used by Compaq Fortran. See the Fortran PowerStation documentation for more detailed information.
Page 51 of 130
The following table summarizes the default output formats for listdirected output with the intrinsic data types: Default Formats for List-Directed Output Data Type BYTE LOGICAL (all) INTEGER(1) INTEGER(2) INTEGER(4) INTEGER(8) REAL(4) REAL(8) COMPLEX(4) COMPLEX(8) CHARACTER
l
Output Format with /fpscomp:noioformat I5 L2 I5 I7 I12 I22 1PG15.7E2 1PG24.15E3 '( ',1PG14.7E2, ', ',1PG14.7E2, ') ' '( ',1PG23.15E3, ', ',1PG23.15E3, ') ' Aw
Output Format with /fpscomp:ioformat I12 L2 I12 I12 I12 I22 1PG16.6E2 1PG25.15E3 '( ',1PG16.6E2, ', ',1PG16.6E2, ') ' '( ',1PG25.15E3, ', ',1PG25.15E3, ') ' Aw
The READ statement: Formatted List-Directed: READ(eunit, * [, iostat]...) Formatted Internal List-Directed: READ(iunit, * [, iostat]...) Formatted Namelist: READ(eunit, nml-group [, iostat]...) If /fpscomp:ioformat is set, the field width values and the list-directed semantics are dictated according to the following sample for real constants (N below):
n
For 1 <= N < 10**7, use F15.6 for single precision or F24.15 for double. For N < 1 or N >= 10**7, use E15.6E2 for single precision or E24.15E3 for double.
Page 52 of 130
See the Fortran PowerStation documentation for more detailed information about the other data types affected.
Unformatted: READ(eunit [, iostat]...) If /fpscomp:ioformat is set, the unformatted file semantics are dictated according to the Fortran PowerStation documentation. Be aware that the file format to read differs from that used by Compaq Fortran. See the Fortran PowerStation documentation for more detailed information.
/fpscomp:[no]ldio_spacing For list-directed output, controls whether a blank is inserted at run-time after a numeric value before a character value (undelimited character string). The default is /fpscomp:noldio_spacing, which conforms to the Fortran 95 standard by inserting a blank after a numeric value before a character value. To request non-standard behavior for compatibility with Microsoft Fortran PowerStation and Visual Fortran releases before Version 6.6, either specify /fpscomp:ldio_spacing or specify /fpscomp:general, which sets /fpscomp:ldio_spacing. /fpscomp:[no]libs Controls whether the library dfport.lib (Portability library) is passed to the compiler and linker. The default is /fpscomp:libs, which passes this library. Specifying /fpscomp:nolibs does not pass this library. /fpscomp:[no]logicals Controls the value used for logical true. Microsoft Fortran PowerStation and Compaq Fortran with the /fpscomp:logical option set uses any non-zero value (default is 1) for true. Compaq Fortran with the /fpscomp:nological option set only looks at the low bit of the value, using a -1 for true. Differences can occur when a logical is stored into an integer. Both use 0 (zero) for false. This affects the results of all logical expressions and affects the return value for the following Fortran features:
l
l l l
The INQUIRE statement specifiers OPENED, IOFOCUS, EXISTS, and NAMED. The EOF intrinsic function. The BTEST intrinsic function. The lexical intrinsic functions LLT, LLE, LGT, and LGE.
/fpscomp:[no]symbols
Page 53 of 130
Adds one or more symbols related to Microsoft Fortran PowerStation to preprocessor and compiler invocations. The symbol currently set by specifying /fpscomp:symbols is _MSFORTRAN_=401. /fpscomp:all and /fpscomp Enable full Microsoft Fortran PowerStation compatibility or /fpscomp: (filesfromcmd,general,ioformat,ldio_spacing,libs,logicals,symbols). /nofpscomp or /fpscomp:none Enables full Compaq Fortran compatibility or /fpscomp: (nofilesfromcmd,nogeneral,noioformat,noldio_spacing,nolibs,nologicals,nosymbols). If you omit /fpscomp, the default is /fpscomp:libs. The /fpscomp and /vms options are not allowed in the same command.
/[no]free
Syntax: /free, /nofree,/4Yf, or /4Nf The /free or /4Yf option specifies that the source file should be interpreted as being in free source format. Equivalent options are as follows:
l
The /nofixed, /free, and /4Yf options are equivalent and request freesource form. The /fixed, /nofree, and /4Nf options are equivalent and request fixedsource form.
In the visual development environment, specify the Source Form in the Fortran Language Compiler Option Category. If you omit /[no]free, /[no]fixed, /4Nf, and /4Yf, the compiler assumes:
l l
Files with an extension of .f90, .F90, or .i90 are free-format source files. Files with an extension of .f, .for, .FOR, or .i are fixed-format files.
/help
Syntax: /help or /?
Page 54 of 130
The /help and /? option display information about the DF command. The option can be placed anywhere on the command line. For a table of DF command options listed alphabetically, see Options List, Alphabetic Order.
/iface
Syntax: /iface[:keyword...] The /iface option determines the type of argument-passing conventions used by your program for general arguments and for hidden-length character arguments. In the visual development environment, specify the Default Calling Conventions and the String Length Argument Passing in the External Procedures Compiler Option Category. The /iface keywords are as follows:
l
The general argument-passing convention keywords are one of: cref, stdref, and default (stdref and default are equivalent). The functions performed by each are described in the following table: /iface:cref Arguments are passed Append @n to names on ia32 systems? Who cleans up stack Var args support? By reference No Caller Yes /iface:default By reference Yes Callee No /iface:stdref By reference Yes Callee No
To specify the convention for passing the hidden-length character arguments, specify /iface:[no]mixed_str_len_arg: /iface:mixed_str_len_arg Requests that the hidden lengths be placed immediately after their corresponding character argument in the argument list, which is the method used by Microsoft Fortran PowerStation.
/iface:nomixed_str_len_arg
Page 55 of 130
Requests that the hidden lengths be placed in sequential order at the end of the argument list, which is the method used by Compaq Fortran for Tru64 UNIX systems by default. When porting mixedlanguage programs that pass character arguments, either this option must be specified correctly or the order of hidden length arguments changed in the source code. If you omit the /iface option, the following is used:
/iface=(default,mixed_str_len_arg)
For more information on argument passing, see Programming with Mixed Languages.
/imsl
Syntax: /imsl The /imsl option ensures that the IMSL libraries will be passed to the linker. Your program still needs to specify the appropriate USE statement(s) and set IMSL environment variables (see Using the Libraries from Visual Fortran). In the visual development environment, specify Use IMSL Math Library in the Library Compiler Option Category.
/[no]include
Syntax: /include[:path...], /noinclude, or /Ipath The /include or /I option specifies one or more additional directories (path) to be searched for module files (USE statement) and include files (INCLUDE statement). In the visual development environment, specify Custom INCLUDE and USE Path in the Preprocessor Compiler Option Category. For all USE statements and for those INCLUDE statements whose file name does not begin with a device or directory name, the directories searched are as follows: 1. The directory containing the first source file or the current directory (depends on whether /assume:source_include was specified).
Page 56 of 130
2. The current default directory where the compilation is taking place 3. If specified, the directory or directories listed in the /include:path or /Ipath option. The order of searching multiple directories occurs within the specified list from left to right 4. The directories indicated in the environment variable INCLUDE To request that the compiler search first in the directory where the source file resides instead of the current directory, specify /assume:source_include. Specifying /noinclude (or /include or /I without a path) prevents searching in the standard directory specified by the INCLUDE environment variable.
/[no]inline
Syntax: /inline[:keyword], /noinline, or /Ob2 The /inline or /Ob2 option allows users to have some control over inlining. Inlining procedures can greatly improve the run-time performance for certain applications. When requesting procedure inlining (or interprocedural optimizations), compile all source files together into a single object file whenever possible. With very large applications, compile as many related source files together as possible. If you compile sources without linking (see the /compile_only option), be sure to also specify the /object[:filename] option to create a single object file. In the visual development environment, specify the Inlining type in the Optimizations Compiler Option Category. The /inline options are: /inline:none or /noinline /inline:manual /inline:size inline:all The /inline options:
l
/inline:speed or /Ob2
/inline:none or /noinline Prevents the inlining of procedures, except for statement functions. This type of inlining occurs if you specify /optimize:0 or /optimize:1 and omit /inline options.
Page 57 of 130
/inline:manual Prevents the inlining of procedures, except for statement functions. This type of inlining occurs if you specify /optimize:2 or /optimize:3 and omit /inline options.
/inline:size Inlines procedures that will improve run-time performance without significantly increasing program size. It includes the types of procedures inlined when you specify /inline:manual.
/inline:speed or /Ob2 Inlines procedures that will improve run-time performance with a significant increase in program size. This type of inlining is available with /optimize:1 or higher. This type of inlining occurs if you specify /optimize:4 or /optimize:5 and omit /inline options.
inline:all Inlines absolutely every call that it is possible to inline while still getting correct code. However, recursive routines will not cause an infinite loop at compile time. This type of inlining is available with /optimize:1 or higher. It includes the types of procedures inlined when you specify other /inline options. Using /inline:all can significantly increase program size and slow compilation speed.
For more detailed information on this option, see Controlling the Inlining of Procedures.
/[no]intconstant
Syntax: /intconstant or /nointconstant The /intconstant option requests that Fortran 77 semantics (type determined by the value) be used to determine the kind of integer constants instead of Fortran 90 default INTEGER type. If you do not specify /intconstant, the type is determined by the default INTEGER type. In the visual development environment, specify Use F77 Integer Constants in the Compatibility Compiler Option Category.
Page 58 of 130
/integer_size
Syntax: /integer_size:size or /4I2 The /integer_size or /4I2 option specifies the size (in bits) of integer and logical declarations, constants, functions, and intrinsics. In the visual development environment, specify the Default Integer Kind in the Fortran Data Compiler Option Category. These options are:
l
/integer_size:16 or /4I2 Makes the default integer and logical variables 2 bytes long. INTEGER and LOGICAL declarations are treated as (KIND=2).
/integer_size:32 Makes the default integer and logical variables 4 bytes long (default). INTEGER and LOGICAL declarations are treated as (KIND=4).
/integer_size:64 Makes the default integer and logical variables 8 bytes long. INTEGER and LOGICAL declarations are treated as (KIND=8).
/[no]keep
Syntax: /keep or /nokeep The /keep option creates one object file for each input source file specified, which may not be desirable when compiling multiple source files. The /keep option does not remove temporary files, which might be created by the FPP preprocessor or the Visual Fortran compiler. If the /keep option is specified, the FPP output files and object files are created in the current directory and retained. The /keep option also affects the number of files that are created and the file names used for these files.
/[no]libdir
Syntax:
Page 59 of 130
/libdir[:keyword], /nolibdir, or /Zl or /Zla The /libdir, /Zl, or /Zla option controls whether library search paths are placed into object files generated by the compiler. Specify one or more of the following options: /libdir:all or /libdir /libdir:none, /nolibdir, or /Zl /libdir:automatic The /libdir options:
l
/libdir:user
/libdir:all or /libdir Requests the insertion of linker search path directives for libraries automatically determined by the DF command driver and for those specified by the cDEC$ OBJCOMMENT LIB source directives. Specifying /libdir:all is equivalent to /libdir:(automatic, user). This is the default.
/libdir:none, /nolibdir, or /Zla Prevents all linker search path directives from being inserted into the object file (neither automatic nor user specified).
/libdir:automatic Requests the insertion of linker search path directives for libraries automatically determined by the DF command driver (default libraries). To prevent the insertion of linker directives for default libraries, specify /libdir:noautomatic or /Zl. In the visual development environment, specify Disable Default Library Search Rules (for /libdir:noautomatic) in the Libraries Compiler Option Category.
/libdir:user Allows insertion of linker search path directives for any libraries specified by the cDEC$ OBJCOMMENT LIB source directives. To prevent the insertion of linker directives for any libraries specified by the OBJCOMMENT directives, specify /libdir:nouser. In the visual development environment, specify Disable OBJCOMMENT Directives (for /libdir:nouser) in the Libraries Compiler Option Category.
/libs
Page 60 of 130
Syntax: /libs [:keyword], /MD, /MDd, /MDs, /ML, /MLd, /MT, /MTd, /MTs, /MW, or /MWs The /libs option controls the type of libraries your application is linked with. The default is /libs:static (same as /libs). In the visual development environment, specify the appropriate options in Use Fortran Run-Time Libraries list (for example, an option with DLL in its name sets /libs:dll) in the Libraries Compiler Option Category. These options are: /libs:dll or /MDs /libs:static or /ML
/libs:qwin or /MW /libs:qwins or /MWs This section also contains jumps to related information about libraries. The /libs options:
l
/libs:dll or /MDs The /libs:dll or /MDs option causes the linker to search for unresolved references in single threaded, dynamic link reference libraries (DLLs). If the unresolved reference is found in the DLL, it gets resolved when the program is executed (during program loading), reducing executable program size. Specifying /libs:dll with /threads is equivalent to /MD. Specifying /libs:dll with /threads and /dbglibs is equivalent to /MDd.
/libs:static or /ML The /libs:static or /ML option requests that the linker searches only in single threaded, static libraries for unresolved references. This is the default. Specifying /libs:static does not request that dynamic link libraries (DLLs), QuickWin, or Standard Graphics libraries be searched. If you use QuickWin or Standard Graphics routines, use /libs:qwin or /libs:qwins. Specifying /libs (with no keyword) is the same as specifying /libs:static. Specifying Specifying Specifying Specifying /libs:static /libs:static /libs:static /libs:static with with with with /nothreads is equivalent to /ML. /nothreads and /dbglibs is equivalent to /MLd. /threads is equivalent to /MT. /threads and /dbglibs is equivalent to /MTd.
/libs:qwin or /MW
Page 61 of 130
Specifying /libs:qwin or /MW requests linking with libraries required of a Fortran QuickWin multi-doc (QuickWin) application.
l
/libs:qwins or /MWs Specifying /libs:qwins or /MWs requests linking with libraries required of a Fortran Standard Graphics (QuickWin single-doc) application.
To request the creation of a dynamic-link library, see /dll. For information about compiling and linking Visual Fortran and Visual C++ programs (and the libraries used), see Specifying Consistent Library Types and Visual Fortran/Visual C++ Mixed-Language Programs. For command-line examples of using the /libs option, see Linking Against Additional Libraries.
/[no]link
Syntax: /link:options or /nolink The /link option (without specifying options) precedes options to be passed to the linker as is (see Linker Options and Related Information). You can also specify the options to be passed to the linker as is using the form: /link:options. To specify additional libraries to be linked on the command line, specify the library name on the DF command line either before or after the /link option, but the Linker option /nodefaultlib must follow the /link option:
DF /compiler-options filename.f90 mylib.lib /link /nodefaultlib
In the visual development environment, you can specify linker options using the Linker tab in the Project menu Settings dialog box. For example, to specify additional libraries to be linked in the visual development environment: 1. In the Project menu, click Settings to display the Project settings dialog box 2. Click the Linker tab
Page 62 of 130
3. Select the General category 4. Type the additional library name to be linked with under Object/Library modules, such as mylib.lib 5. Click OK when done The /nolink option suppresses linking and forces an object file to be produced even if only one program is compiled. Any options specified after the /nolink option are ignored. If you encounter linker errors, see Linker Diagnostic Messages and Error Conditions.
/[no]list
Syntax: /list[:file], /nolist, or /Fsfile The /list or /Fs option creates a listing of the source file with compile-time information appended. To name the source listing file, specify file. If you omit the /list or /Fs options (or specify /nolist), no listing file is created. In the visual development environment, specify Source Listing in the Listing File Compiler Option Category. When a diagnostic message is displayed, the listing file contains a column pointer (such as .....1) that points to the specific part of the source line that caused the error. To specify the content of the listing file, see /show. To request a listing with Assembly instructions, see /asmfile. To request a listing with information about optimizations, see /annotations. The name of the listing file is the same as the source file (unless specified by file), with the extension .LST (unless the extension is specified by file). If multiple object files are created, multiple listing files are usually created. For example, if you specify multiple source files with the /compile_only and /list options without a named object file (/object:file), multiple files are created. If you specify multiple source files with the /list, /compile_only, and /object:file, a single listing file is created. For command-line examples, see Generating a Listing File.
/[no]logo
Page 63 of 130
Syntax: /nologo or /logo The /nologo option suppresses the copyright notice displayed by the compiler and linker. This option can be placed anywhere on the command line.
/[no]machine_code
Syntax: /machine_code or /nomachine_code The /machine_code option requests that a machine language representation be included in the listing file. The /machine_code option is a synonym for /show:code. In the visual development environment, specify Source Listing Options, Machine Code in the Listing File Compiler Option Category. This option is ignored unless you specify /list[:file].
/[no]map
Syntax: /map[:file], /nomap, or /Fmfile The /map or /Fm option controls whether or not a link map is created. To name the map file, specify file. In the visual development environment, in the Project menu Settings dialog box: 1. Click the Linker tab 2. Select the General category 3. Click the Generate mapfile option check box If you omit /map or /Fm, a map file is not created. The link map is a text file (see the Linker option /MAP).
/math_library
Syntax: /math_library:keyword
Page 64 of 130
The /math_library option specifies whether argument checking of math routines is done on ia32 systems. This option is ignored on ia64 systems. In the visual development environment, specify the Math Library in the Optimizations (or Code Generation) Compiler Option Category. The /math_library options are: /math_library:fast and /math_library:check:
l
/math_library:fast On ia32 systems, /math_library:fast improves performance by not checking the arguments to the math routines. Using /math_library:fast makes tracing the cause of unexpected exceptional values results difficult. Specifying /math_library:fast does not change the accuracy of calculated floating-point numbers.
/math_library:check On ia32 systems, /math_library:check validates the arguments to and results from calls to the Fortran math routines. This provides slower runtime performance than /math_library:fast, but with earlier detection of exceptional values. This is the default on ia32 systems.
/[no]module
Syntax: /module[:path] or /nomodule The /module option controls where the module files (extension .MOD) are placed. If you omit this option (or specify /nomodule), the .MOD files are placed in the directory where the source file being compiled resides. When /module:path is specified, the path specifies the directory location where the module files will be placed. In the visual development environment, specify the Module Path in the Preprocessor Compiler Option Category. When /module is entered without specifying a path, it is interpreted as a request to place the MOD files in the same location that the object is being created. Should a path be specified on the /object option, that location would also be used for the MOD files. You need to ensure that the module files are created before they are referenced when using the DF command (see Compile With Appropriate Options and Multiple Source Files).
Page 65 of 130
/names
Syntax: /names:keyword, /GNa, /GNl, or /GNu The /names option specifies how source code identifiers and external names are interpreted and the case convention used for external names. This naming convention applies whether names are being defined or referenced. The default is /names:uppercase (same as /GNu). In the visual development environment, specify the Name Interpretation in the External Procedures or the Fortran Language Compiler Option Category. The /names options are:
l
/names:as_is or /GNa Causes the compiler to: Distinguish between uppercase and lowercase letters in source code identifiers (treat uppercase and lowercase letters as different). Distinguish between uppercase and lowercase letters in external names. /names:lowercase or /GNl Causes the compiler to: Not distinguish between uppercase and lowercase letters in source code identifiers (treat lowercase and uppercase letters as equivalent). Force all letters to be lowercase in external names. /names:uppercase or /GNu (default) Causes the compiler to: Not distinguish between uppercase and lowercase letters in source code identifiers (treat lowercase and uppercase letters as equivalent). Force all letters to be uppercase in external names.
Instead of using the /names compiler option, consider using the cDEC$ ALIAS directive for the specific name needed.
/nodefine
Syntax: /nodefine The /nodefine option requests that all symbols specified by the accompanying /define:symbols option apply only to the Fortran preprocessor, FPP, and are not available to conditional compilation directives (such as the IF
Page 66 of 130
Directive Construct). For example, the following command defines the symbol release as 1, which is available only to FPP:
DF /fpp /define:release /nodefine
If you specify /define:symbols and omit /nodefine, symbols specified by /define:symbols are available to both FPP and conditional compilation directives. In the visual development environment, specify the Predefined Preprocessor Symbols to FPP Only in the Preprocessor Compiler Option Category. For more information on FPP, see /fpp.
/[no]object
Syntax: /object[:filename], /noobject, or /Fofilename The /object or /Fo option names the object file filename. Specify /noobject to prevent creation of an object file. The default is /object, where the file name is the same as the first source file with a file extension of .OBJ. If you omit /compile_only (or /c) and specify /object:filename or /Fofilename, the /object option names the object file filename. If you specify /object:filename or /Fofilename and specify the /compile_only option, the /object or /Fo option causes multiple Fortran input files (if specified) to be compiled into a single object file. This allows interprocedural optimizations to occur at higher optimization levels, which usually improves run-time performance. For information on where module files are placed, see /module[:path].
/[no]optimize
Syntax: /optimize[:level], /nooptimize, /Od, /Ox, or /Oxp The /optimize option controls the level of optimization performed by the compiler. To provide efficient run-time performance, Visual Fortran increases compile time in favor of decreasing run time. If an operation can be performed, eliminated, or simplified at compile time, the compiler does so rather than have it done at run time. Also, the size of object file usually increases when certain optimizations occur (such as with more loop unrolling and more inlined
Page 67 of 130
In the visual development environment, specify the Optimization Level in the General or Optimizations Compiler Option Category. The /optimize options are: /optimize:0 or /Od /optimize:2 /optimize:1 /optimize:3
/optimize:0 or /Od Disables nearly all optimizations. This is the default if you specify /debug (with no keyword). Specifying this option causes certain /warn options to be ignored. The /optimize:0 option is the default in the the visual development environment for a debug configuration. Specifying /Od sets the /optimize:0 and /math_library:check options.
/optimize:1 Enables local optimizations within the source program unit, recognition of common subexpressions, and expansion of integer multiplication and division (using shifts).
/optimize:2 Enables global optimization. This includes data-flow analysis, code motion, strength reduction and test replacement, split-lifetime analysis, and instruction scheduling. Specifying /optimize:2 includes the optimizations performed by /optimize:1.
/optimize:3 Enables additional global optimizations that improve speed (at the cost of extra code size). These optimizations include:
Loop unrolling, including instruction scheduling Code replication to eliminate branches Padding the size of certain power-of-two arrays to allow more efficient cache use (see Use Arrays Efficiently)
Page 68 of 130
/optimize:4, /Ox, and /Oxp Enables interprocedure analysis and automatic inlining of small procedures (with heuristics limiting the amount of extra code). Specifying /optimize:4 includes the optimizations performed by /optimize:1 /optimize:2, and /optimize:3. For the DF command, /optimize:4 is the default for the command line unless you specify /debug (with no keyword). In the the visual development environment for a release configuration, the default is /optimize:4. Specifying /Ox sets: /optimize:4, /math_library:fast, and /assume:nodummy_aliases. Specifying /Oxp sets: /optimize:4, /math_library:check, /assume:nodummy_aliases, and /fltconsistency.
/optimize:5 On ia32 systems, activates the loop transformation optimizations (also set by /transform_loops). On ia64 systems, activates the loop transformation optimizations (also set by /transform_loops) and the software pipelining optimization (also set by /pipeline (ia64 only)):
The loop transformation optimizations are a group of optimizations that apply to array references within loops. These optimizations can improve the performance of the memory system and can apply to multiple nested loops. Loop transformation optimizations include loop blocking, loop distribution, loop fusion, loop interchange, loop scalar replacement, and outer loop unrolling. You can specify loop transformation optimizations without software pipelining (see /[no]transform_loops).
The software pipelining optimization (ia64 systems only) applies instruction scheduling to certain innermost loops, allowing instructions within a loop to "wrap around" and execute in a different iteration of the loop. This can reduce the impact of long-latency operations, resulting in faster loop execution. Software pipelining also enables the prefetching of data to reduce the impact of cache misses. You can specify software pipelining without loop transformation optimizations (see /[no]pipeline (ia64 only)).
To determine whether using /optimize:5 benefits your particular program, you should compare program execution timings for the same program (or
Compiler and Linker Options subprogram) compiled at levels /optimize:4 and /optimize:5. Specifying /optimize:5 includes the optimizations performed by /optimize:1, /optimize:2, /optimize:3, and /optimize:4.
Page 69 of 130
For detailed information on these optimizations, see Optimization Levels: the /optimize Option. For information about timing your program, see Analyze Program Performance. To compile your application for efficient run-time performance, see Compile With Appropriate Options and Multiple Source Files.
/[no]pad_source
Syntax: /pad_source or /nopad_source For fixed-form source files, the /pad_source option requests that source records shorter than the statement field width are to be padded with spaces on the right out to the end of the statement field. This affects the interpretation of character and Hollerith literals that are continued across source records. In the visual development environment, specify the Pad Fixed-Form Source Records in the Fortran Language Compiler Option Category. The default is /nopad_source, which causes a warning message to be displayed if a character or Hollerith literal that ends before the statement field ends is continued onto the next source record. To suppress this warning message, specify the /warn:nousage option. Specifying /pad_source can prevent warning messages associated with /warn:usage.
/[no]pdbfile
Syntax: /pdbfile[:filename], /nopdbfile, or /Fdfilename The /pdbfile or /Fd option indicates that any debug information generated by the compiler should be to a program database file, filename.PDB. If you omit filename, the default file name used is df60.pdb. In the visual development environment, specify Use Program Database for Debug Information (and optionally specify the Program Database .PDB Path) in
Page 70 of 130
When full debug information is requested (/debug:full, /debug, or equivalent), the debug information is placed in the PDB file (unless /nopdbfile is specified). The compiler places debug information in the object file if you specify /nopdbfile or omit both /pdbfile and /debug:full (or equivalent).
Specify /optimize:5 with /notransform_loops. Specify /pipeline with /optimize:4, /optimize:3, or /optimize:2. This optimization is not performed at optimization levels below /optimize:2.
To determine whether using /pipeline benefits your particular program, you should time program execution for the same program (or subprogram) compiled with and without software pipelining (such as with /pipeline and /nopipeline). For certain programs that contain loops that exhaust available registers, longer execution times may result with /optimize:5, requiring use of /unroll:count to
Page 71 of 130
limit loop unrolling. The /pipeline option applies only to ia64 systems. For more information, see Software Pipelining.
/preprocess_only
Syntax: /preprocess_only When used with the /fpp option, the /preprocess_only option runs only the FPP preprocessor and puts the result for each source file in a corresponding .i or .i90 file. The .i or .i90 file does not have line numbers (#) in it. You need to specify /fpp with this option.
/real_size
Syntax: /real_size:size or /4R8 The /real_size or /4R8 option controls the size (in bits) of REAL and COMPLEX declarations, constants, functions, and intrinsics. In the visual development environment, specify the Default Real Kind in the Fortran Data Compiler Option Category. The /real_size options are:
l
/real_size:32 Defines REAL declarations, constants, functions, and intrinsics as REAL (KIND=4) (SINGLE PRECISION). It also defines COMPLEX declarations, constants, functions, and intrinsics as COMPLEX(KIND=4) (COMPLEX). This is the default.
/real_size:64 or /4R8 Defines REAL declarations, constants, functions, and intrinsics as REAL (KIND=8) (DOUBLE PRECISION). It also defines COMPLEX declarations, constants, functions, and intrinsics as COMPLEX(KIND=8).
Specifying /real_size:64 causes intrinsic functions to produce a REAL(KIND=8) or COMPLEX(KIND=8) result instead of a REAL(KIND=4) or COMPLEX(KIND=4) result, unless the argument is explicitly typed as REAL(KIND=4) or COMPLEX (KIND=4), including CMPLX, FLOAT, REAL, SNGL, and AIMAG. For instance, references to the CMPLX intrinsic produce DCMPLX results (COMPLEX(KIND=8)), unless the argument to CMPLX is explicitly typed as REAL(KIND=4), REAL*4, COMPLEX(KIND=4), or COMPLEX*8. In this case the resulting data type is
Page 72 of 130
/[no]recursive
Syntax: /recursive or /norecursive The /recursive option compiles all procedures (functions and subroutines) for possible recursive execution. Specifying the /recursive option sets the /automatic option. The default is /norecursive. In the visual development environment, specify Enable Recursive Routines in the Code Generation Compiler Option Category.
/[no]reentrancy
Syntax: /reentrancy[:keyword] or /noreentrancy The /reentrancy or /reentrancy:threads option requests that the compiler generate reentrant code that supports a multithreaded application. In the visual development environment, specify the Enable Reentrancy Support or Disable Reentrancy Support in the Libraries Compiler Option Category. If you omit /reentrancy, /reentrancy:threads, or /threads, /reentrancy:none (same as /noreentrancy) is used. Specifying /threads sets /reentrancy:threads, since multithreaded code must be reentrant.
/[no]show
Syntax: /show:keyword... or /noshow The /show option specifies what information is included in a listing. In the visual development environment, specify the Source Listing Options in the Listing File Compiler Option Category. The /show keywords are:
Page 73 of 130
/show:code /show:map
/show:include /show:nomap
/show:code Includes a machine-language representation of the compiled code in the listing file. The default is /show:nocode. The /show:code and /machine_code options are equivalent.
/show:include Lists any text file included in the source file (unless that source is included using the INCLUDE 'filespec /NOLIST' syntax; see the /vms option). The default is /show:noinclude.
/source
Syntax: /source:file or /Tffile The /source or /Tf option indicates that the file is a Fortran source file with a
Page 74 of 130
non-standard file extension (not one of .F, .FOR, or .F90) that needs to be compiled. The default for any file that does not have an extension of .F90 or .f90 is to be a fixed-format Fortran file.
/[no]stand
Syntax: /stand[:keyword], /nostand, /4Ys, or /4Ns The /stand option issues compile-time messages for language elements that are not standard in the Fortran 90 or Fortran 95 language that can be identified at compile-time. In the visual development environment, specify the Fortran Standards Checking in the Compilation Diagnostics Compiler Option Category. The /stand options are:
l
/stand:f90 or /4Ys Specifies that diagnostic messages be generated with a warning-level severity (allows an object file to be created) for extensions to the Fortran 90 standard.
/stand or /stand:f95 Specifies that diagnostic messages be generated with a warning-level severity (allows an object file to be created) for extensions to the Fortran 95 standard.
/stand:none, /nostand, or /4Ns Specifies that no messages be issued for language elements that are not standard in the Fortran 90 or Fortran 95 language. The same effect occurs if you omit the /stand, or /stand:keyword, /warn:stderrors, or /4Ys options.
Specify /warn:stderrors to request that diagnostic messages be generated with an error-level severity (instead of warning) to prevent an object file from being created. Specifying /stand issues warning messages for:
l l
Obsolescent and deleted features specified by the Fortran standard. Syntax extensions to the Fortran 95 standard. Syntax extensions include
Page 75 of 130
nonstandard statements and language constructs. Fortran 95 standard-conforming statements that become nonstandard due to the way in which they are used. Data type information and statement locations are considered when determining semantic extensions. For fixed-format source files, lines that use tab formatting.
Source statements that do not conform to Fortran language standards are detected by the compiler under the following circumstances:
l l
The statements contain ordinary syntax and semantic errors. A source program containing nonconforming statements is compiled with the /stand or /check options.
Given these circumstances, the compiler is able to detect most instances of nonconforming usage. It does not detect all instances because the /stand option does not produce checks for all nonconforming usage at compile time. In general, the unchecked cases of nonconforming usage arise from the following situations:
l
The standard violation results from conditions that cannot be checked at compile time. The compile-time checking is prone to false alarms.
Most of the unchecked cases occur in the interface between calling and called subprograms. However, other cases are not checked, even within a single subprogram. The following items are known to be unchecked:
l l
Use of a data item prior to defining it Use of the SAVE statement to ensure that data items or common blocks retain their values when reinvoked Association of character data items on the right and left sides of character assignment statements Mismatch in order, number, or type in passing actual arguments to subprograms with implicit interfaces Association of one or more actual arguments with a data item in a common block when calling a subprogram that assigns a new value to one or more of the arguments
/[no]static
Syntax: /static or /nostatic The /static option is the same as the /noautomatic option. The default is /static,
Page 76 of 130
which causes all local variables to be statically allocated. The /nostatic option is the same as /automatic. In the visual development environment, specify /nostatic as Variables Default to Automatic in the Fortran Data Compiler Option Category. If you specify /recursive, the /automatic option is set.
/[no]syntax_only
Syntax: /syntax_only or /nosyntax_only The /syntax_only option requests that only the syntax of the source file be checked. If the /syntax_only option is specified, code generation is suppressed. The default is /nosyntax_only.
/[no]threads
Syntax: /threads or /nothreads The /threads option requests linking with multithreaded libraries, which creates a multithreaded program or DLL. If you specify /threads, this sets the /reentrancy option. In the visual development environment, specify one of the Multi-thread library options in Use Fortran Run-Time Libraries list in the Libraries Compiler Option Category. The default is /nothreads, which links with single-threaded libraries to create a single-threaded program or DLL. Related options that control the libraries used during linking include:
l l l
/[no]traceback
Syntax: /traceback or /notraceback or /Zt
Page 77 of 130
The /traceback option requests that the compiler generate extra information in the object file that allows the display of source file traceback information at run time when a severe error occurs. Specifying /traceback:
l
Provides source file, routine name, and line number correlation information in the text that is displayed when a severe error occurs. Will increase the size of the executable program, but has no impact on runtime execution speeds.
For the DF command line and for a release configuration in the visual development environment, the default is /notraceback. For a debug configuration in the visual development environment, the default is /traceback. If traceback is not specified, the displayed call stack hexadecimal addresses (program counter trace) displayed when a severe error occurs do not list the source file name, routine name, and line number. However, advanced users can locate the cause of the error using a .MAP file (linker option /map) and the hexadecimal addresses of the stack displayed when a severe error occurs (see Using Traceback Information). In the visual development environment, specify Generate Traceback Information in the Run-Time Compiler Option Category. The /traceback option functions independently of the /debug option. If you request traceback, you should also disable incremental linking. For a Debug configuration in the visual development environment for a new project, specifying Traceback turns off incremental linking. When using the command line, specifying /traceback sets /link /incremental:no. You can disable incremental linking either in the Link tab in the Project Settings dialog box or specify DF /link /incremental:no on the command line. If you omit /traceback (or /Zt), /notraceback is used. For information about locating run-time errors with traceback information, see Using Traceback Information. For request traceback information at any time by calling the TRACEBACKQQ routine, see Obtaining Traceback Information with TRACEBACKQQ. To disable the stack traceback report for severe errors, set the FOR_DISABLE_STACK_TRACE environment variable (see Using Traceback Information).
Page 78 of 130
/[no]transform_loops
Syntax: /transform_loops or /notransform_loops The /transform_loops (or /optimize:5) option activates a group of loop transformation optimizations that apply to array references within loops. These optimizations can improve the performance of the memory system and usually apply to multiple nested loops. The loops chosen for loop transformation optimizations are always counted loops (which include DO or IF loops), but not uncounted DO WHILE loops). In the visual development environment, specify the Apply Loop Transformation Optimizations in the Optimizations Compiler Option Category. Conditions that typically prevent the loop transformation optimizations from occurring include subprogram references that are not inlined (such as an external function call), complicated exit conditions, and uncounted loops. The types of optimizations associated with /transform_loops include the following:
l l l l l l
Loop blocking Loop distribution Loop fusion Loop interchange Loop scalar replacement Outer loop unrolling
The loop transformation optimizations are a subset of optimizations activated by /optimize:5. On ia32 systems, instead of specifying /transform_loops, you can specify /optimize:5. On ia64 systems, instead of specifying both /pipeline and /transform_loops, you can specify /optimize:5. On ia64 systems, to specify loop transformation optimizations without software pipelining, do one of the following:
l l
Specify /optimize:5 with /nopipeline. Specify /transform_loops with /optimize:4, /optimize:3, or /optimize:2. This optimization is not performed at optimization levels below /optimize:2.
To determine whether using /transform_loops benefits your particular program, you should time program execution for the same program (or subprogram) compiled with and without loop transformation optimizations (such as with /transform_loops and /notransform_loops).
Page 79 of 130
/tune
Syntax: /tune:keyword The /tune option specifies the type of processor-specific machine-code instruction tuning for implementations of the processor architecture in use. Tuning for a specific implementation can improve run-time performance; it is also possible that code tuned for a specific processor may run slower on another processor. Regardless of the /tune:keyword option you use, the generated code runs correctly on all implementations of the processor architecture. If you omit /tune:keyword, /tune:generic is used. In the visual development environment, specify the Optimize For in the Optimizations Compiler Option Category. For ia32 (32-bit Intel and AMD) systems, the /tune keywords are:
l
/tune:generic Generates and schedules code (sometimes called blended code) that will execute well for all ia32 systems. This provides generally efficient code for those applications where all ia32 processor generations are likely to be used. This is the default.
/tune:host Generates and schedules code optimized for the processor type in use on the processor system being used for compilation.
/tune:pn1 Generates and schedules code optimized for the Pentium (586) processor systems. The pn1 keyword replaces the p5 keyword (specifying /tune:pn1 and /tune:p5 are equivalent).
/tune:pn2 Generates and schedules code optimized for Pentium Pro, Pentium II, and AMD K6 processor systems. The pn2 keyword replaces the p6 keyword (specifying /tune:pn2 and /tune:p6 are equivalent).
Page 80 of 130
/tune:k6 Generates and schedules code optimized for AMD K6, Pentium Pro, and Pentium II processor systems (/tune:pn2 and /tune:k6 are the same).
/tune:pn3 Generates and schedules code optimized for Pentium III, AMD K6_2, and AMD K6_III processor systems. The pn3 keyword replaces the p6p keyword (specifying /tune:pn3 and /tune:p6p are equivalent).
/tune:k6_2 Generates and schedules code optimized for AMD K6_2 and AMD K6_III processor systems.
/tune:k7 Generates and schedules code optimized for AMD Athlon processor systems.
/tune:pn4 Generates and schedules code optimized for Pentium IV processor systems.
Specifying /fast sets /tune:host. For ia64 (64-bit) systems, specify either /tune:generic or /tune:host (this option is not yet fully implemented for ia64 systems). The architecture selected by /architecture (code generation option) is the basis for the architecture used by the /tune keyword. For example, you cannot specify a /tune keyword for an architecture older than the one selected by /architecture. For more information about this option, see Requesting Optimized Code for a Specific Processor Generation. For information about timing program execution, see Analyze Program Performance. To control the processor-specific type of machine-code instructions being generated, see the /architecture:keyword option.
/undefine
Page 81 of 130
Syntax: /undefine:symbol The /undefine option removes any initial definition of symbol for the FPP preprocessor.
/unroll
Syntax: /unroll:count For higher optimization levels, the /unroll option allows you to specify how many times loops are unrolled. If the /unroll option is not specified, the optimizer will choose an unroll amount that minimizes the overhead of prefetching while also limiting code size expansion. In the visual development environment, specify the Loop Unroll Count in the Optimizations Compiler Option Category. If the /optimize:3, /optimize:4 (or equivalent), or /optimize:5 options are specified, loop unrolling occurs. The count should be an integer in the range 0 to 16. A count value of 0 is used to indicate that the compiler should determine how many times a loop is unrolled (default). The compiler attempts to unroll certain innermost loops, minimizing the number of branches and grouping more instructions together to allow efficient overlapped instruction execution (instruction pipelining). The best candidates for loop unrolling are innermost loops with limited control flow. On ia32 systems, specifying a higher value may improve run-time performance of certain applications. For more information, see Loop Unrolling. You can specify the number of times a specific loop is unrolled by using the cDEC$ UNROLL Directive to Control Loop Unrolling.
/[no]vms
Syntax: /vms or /novms The /vms option causes the run-time system to provide functions like Compaq Fortran 77 for OpenVMS VAXTM Systems (previously called VAX FORTRANTM).
Page 82 of 130
In the visual development environment, specify Enable VMS Compatibility in the Compatibility Compiler Option Category. The /vms option:
l
In the absence of other options, sets the following command-line defaults: /check:format, /check:output_conversion, /static, /norecursive, and /names:lowercase. When /ccdefault:default is also specified, /vms changes the default carriage control to FORTRAN if the file is formatted and the unit is connected to a terminal. Allows use of the DELETE statement for relative files. When a record in a relative file is deleted, the first byte of that record is set to a known character (currently '@'). Attempts to read that record later result in ATTACCNON errors. The rest of the record (the whole record when /vms is not set) is set to nulls for unformatted files and spaces for formatted files. When an ENDFILE is performed on a sequential unit, an actual 1-byte record containing a Ctrl+D (04 hex) is written to the file. When you omit /vms, an internal ENDFILE flag is set and the file is truncated. The /vms option does not affect ENDFILE on relative files; the file is truncated. Changes certain OPEN statement BLANK keyword defaults. Changes the default interpretation from BLANK='NULL' to BLANK='ZERO' for an implicit open or internal file OPEN. For an explicit OPEN, the default is always BLANK='NULL'. Changes certain OPEN statement effects. If the CARRIAGECONTROL is defaulted, the file is formatted, and the unit is connected to a terminal, then the carriage control defaults to FORTRAN. Otherwise, it defaults to LIST. The /vms option affects the record length for relative organization files. The buffer size is increased by 1 to accommodate the deleted record character. LIST and /NOLIST are recognized at the end of the file specification to the INCLUDE statement at compile time. If you specified /vms and if the file specification does not include the directory path, the current working directory is used as the default directory path. If you omitted /vms, the directory path is where the file that contains the INCLUDE statement resides. Changes internal file writes using list-directed I/O. A list-directed write to an internal file results in removal of the first character from the first element; the field length is decremented accordingly. The run-time direct access READ routine checks the first byte of the retrieved record. If this byte is '@' or NULL ('\0'), then ATTACCNON is returned. The run-time sequential access READ routine checks to see if the record it just read is 1 byte long and contains a Ctrl+D (04 hex) or a Ctrl+Z (1A hex). If this is true, it returns EOF.
Page 83 of 130
/[no]warn
Syntax: /warn[:keyword...]), /nowarn, /4Yd, /4Nd, /4Ys, /W0, /W1, or /WX The /warn option instructs the compiler to generate diagnostic messages for defined classes of additional checking that can be performed at compile-time. It also can change the severity level of issued compilation messages. In the visual development environment, specify the Warning Level (/warn:nogeneral, default, or /warn:error) in the General or the Compiler Diagnostic Compiler Option Category. Specify individual Warning Options in the Compiler Diagnostic Compiler Option Category. The /warn options and their visual development environment names are: /warn:noalignments (Data Alignment) /warn:declarations or /4Yd (Undeclared Variables) /warn:nofileopt (Inform when Compiling Files Separately) /warn:argument_checking (Argument Mismatch) /warn:errors or /WX (Warning Level: Errors) /warn:nogeneral (Warning Level: Ignore)
/warn:ignore_loc /warn:stderror or /4Ys (Inform When Removing %LOC from an (Treat Fortran Standard Warnings as Argument) Errors) /warn:truncated_source (Truncated Source) /warn:nouninitialized (Uninitialized Variables) /warn:nousage (Usage) /warn:none, /nowarn, or /W0 If you omit /warn, the defaults are:
l
For the DF command: /warn: (alignments,noargument_checking,nodeclarations,noerrors,fileopts, general,noignore_loc,nostderrors,notruncated_source,uncalled,uninitialized,nounused, In the visual development environment for a debug configuration, the
Page 84 of 130
default of noargument_checking changes to argument_checking. In the visual development environment for a release configuration, the default of fileopts changes to nofileopts. For the FL32 command: /warn: (alignments,argument_checking,nodeclarations,noerrors,nofileopts, general,noignore_loc,nostderrors,notruncated_source,uncalled,uninitialized,nounused,
/warn:noalignments Suppresses warning messages for data that is not naturally aligned. The default is /warn:alignments.
/warn:argument_checking Enables warnings about argument mismatches between callers and callees, when compiled together. The default is /warn:noargument_checking, except in the visual development environment for a debug configuration, where the default is /warn:argument_checking.
/warn:declarations or /4Yd Issues an error message for any undeclared symbols. This option makes the default type of a variable undefined (IMPLICIT NONE) rather than using the default Fortran rules. The default is /warn:nodeclarations or /4Nd.
/warn:errors or /WX Changes the severity of all warning diagnostics into error diagnostics. The default is /warn:noerrors. Specifying /warn:errors (or /WX) sets /warn:stderrors.
/warn:nofileopt Suppresses the display of an informational-level diagnostic message when compiling multiple files separately, which can prevent interprocedure optimizations. The default is /warn:fileopt (displays the message: Some interprocedural optimizations may be disabled when compiling in this mode), except in the visual development environment for a release configuration, where the default is /warn:nofileopt.
/warn:nogeneral Suppresses all informational-level and warning-level diagnostic messages from the compiler. The default is /warn:general or /W1.
Page 85 of 130
/warn:ignore_loc Requests that the compiler issues warnings when %LOC is stripped from an argument. The default is /warn:noignore_loc (does not issue a warning for this condition).
/warn:stderrors or /4Ys Requests Fortran 90 standards checking (see /stand) with error-level compilation messages instead of warning-level messages. Specifying /warn:stderrors sets /stand:f90 and is equivalent to /4Ys. Specifying /warn:stderrors with /stand:f95 requests error-level messages for extensions to the proposed Fortran 95 standard. Specifying /warn:errors sets /warn:stderrors. The default is /warn:nostderrrors.
/warn:truncated_source Requests that the compiler issue a warning diagnostic message when it reads a source line with a statement field that exceeds the maximum column width in fixed-format source files. The maximum column width for fixed-format files is column 72 or 132, depending whether the /extend_source option was specified. This option has no effect on truncation; lines that exceed the maximum column width are always truncated. This option does not apply to freeformat source files. The default is /warn:notruncated_source.
/warn:nouncalled Suppresses the compiler warning diagnostic message when a statement function is never called. The default is /warn:uncalled.
/warn:nouninitialized Suppresses warning messages for a variable that is used before a value was assigned to it. The default is /warn:uninitialized.
/warn:unused Requests warning messages for a variable that is declared but never used. The default is /warn:nounused.
/warn:nousage
Page 86 of 130
Suppresses warning messages about questionable programming practices and the use of intrinsic functions that use a two-digit year (year 2000). The questionable programming practices, although allowed, often are the result of programming errors. For example, /warn:usage detects a continued character or Hollerith literal whose first part ends before the statement field ends and appears to end with trailing spaces. The default is /warn:usage. The /pad_source option can prevent warning messages from /warn:usage.
l
/warn:all or /warn Requests all possible warning messages, but does not set /warn:errors or /warn:stderrors. To enable all the additional checking to be performed and force the severity of the diagnostics to be severe enough to not generate an object file, specify /warn:(all,errors) or /warn:(all,stderrors).
For example, the following command requests the argument_checking and declarations keywords and accepts the defaults for the other /warn keywords:
df /warn:(argument_checking,declarations) testfile.f90
/[no]watch
Syntax: /watch[:keyword] or /nowatch The /watch option requests the display of processing information to the console output window. The default is /watch:source. You can request the display of the passes (compiler, linker) with their respective command arguments and/or the input and output files by specifying any of the following /watch keywords: /watch:cmd /watch:source
/watch:cmd Displays the passes (compiler, linker) with the respective command arguments.
Page 87 of 130
/watch:source Displays the names of sources file(s) being processed. Source file names are listed one per line. This is the default. /watch:all or /watch Requests /watch:(cmd, source). This displays both pass information and source file names (verbose output). /nowatch or /watch:none Requests /watch:(nocmd, nosource).
/what
Syntax: /what The /what option displays Visual Fortran version number information.
/winapp
Syntax: /winapp or /MG The /winapp or /MG option requests the creation of a graphics or Fortran Windows application and links against the most commonly used libraries. In the visual development environment, specify the Use Common Windows Libraries in the Libraries Compiler Option Category. The following related options request libraries:
l l l l
For information on Fortran Windows Applications, including requesting additional link libraries with the FULLAPI.F90 file, see Creating Windows Applications.
Page 88 of 130
When using the DF command line, specify linker options after the /LINK option. For example:
DF file.f90 file.lib /LINK /NODEFAULTLIB
l
The LINK command line. You can specify linker options and libraries with the LINK command. For example:
LINK file.obj file.lib /NODEFAULTLIB
Within the Microsoft visual development environment, in the Project menu, Settings dialog box. You can specify linker options and libraries by using the Linker tab in the Project menu, Settings dialog box.
The following table describes the Linker options. Linker Option /ALIGN /BASE /COMMENT Function Specifies the alignment of each section within the linear address space of the program. Sets a base address for the program, overriding the default location. Inserts a comment string into the header of an executable file or DLL, after the array of section headers. Creates debugging information for the executable file or DLL. Generates debugging information in one of three ways: Microsoft format (CV), COFF format, or both. Passes a module-definition (.DEF) file to the linker. Adds one or more libraries to the list of libraries that LINK searches when resolving references. Controls the delayed loading of DDLs. Causes delayed loading of DLLs.
Compiler and Linker Options /DLL /DRIVER /ENTRY /EXETYPE /EXPORT /FIXED /FORCE
Page 89 of 130 Builds a DLL as the main output file when using the command line. Used to build a Windows NT 4 or Windows 2000 kernel mode driver. Sets the starting address for an executable file or DLL. Used when building a virtual device driver (requested by using the /VXD option). Exports a function from your program. Tells the operating system to load the program only at its preferred base address. Informs the linker to create a valid executable file or DLL even if a symbol is referenced but not defined or is multiply defined. Sets the size of the heap in bytes. Sets the name for the import library that LINK creates when it builds a program that contains exports. Does not apply to Visual Fortran. Informs the linker to add a specified symbol to the symbol table. Controls how the linker handles incremental linking.
/HEAP /IMPLIB
/LARGEADDRESSAWARE Informs the linker that the application can handle addresses larger than 2 gigabytes. /LIBPATH /LINK50COMPAT /MACHINE /MAP Overrides the environment library path. Generates import libraries in the old (Visual C++ version 5.0) format for backward compatibility. Specifies the target platform for the program. Informs the linker to generate a mapfile. You can also specify the file name.
Page 90 of 130 Informs the linker to include the specified information in a map file (requested by /MAP). Combines the first section with the second section and names the resulting section. Informs the linker to remove all default libraries from the list of libraries it searches when resolving external references. If you specify one or more libraries, the linker only ignores the libraries you have named. Prevents LINK from linking a reference to _main into the DLL. Prevents the display of the copyright message and version number. It also suppresses echoing of command files. Controls the optimizations LINK performs during a build. Lets you perform optimization by telling LINK to place certain packaged functions into the image in a predetermined order. Overrides the default name and location of the image file that LINK creates. Controls how the linker produces debugging information. Controls which Program Database (PDB) is used to store the debug type information. Creates an output file that can be used with the profiler. Sets the checksum in the header of an executable file. Changes the attributes of a section, overriding the attributes set when the .OBJ file for the section was compiled. Sets the size of the stack in bytes.
/NOENTRY /NOLOGO
/OPT /ORDER
/STACK
Page 91 of 130 Attaches an MS-DOS stub program to a Win32 program. Tells the operating system how to run the executable file. Informs the operating system to first copy the linker output to a swap file, and then run the image from there (Windows NT 4 or Windows 2000). Sends information about the progress of the linking session to the Output window. Informs the linker to put a version number in the header of the executable file or DLL. Creates a virtual device driver (VxD). Determines the output of LINK warnings. Adds the WS_AGGRESSIVE attribute to your application's image.
This table lists the Linker options with the equivalent Microsoft visual development environment category if one is available. Options not listed are usually command-line only and can be entered in the "Common Options" text box of the Project ... Settings dialog box. For instructions on how to work with the Microsoft visual development environment, see Building Programs and Libraries. Linker Option /ALIGN /BASE /DEBUG /DEBUGTYPE /DEF /DEFAULTLIB /ENTRY /FORCE Visual Development Environment Category Command-line only Output Category Debug Category Debug Category Command-line only Command-line only Output Category Customize Category
Compiler and Linker Options /INCLUDE /INCREMENTAL /LIBPATH /MAP /NODEFAULTLIB /NOLOGO /OUT /PDB /PDBTYPE /PROFILE /STACK /VERBOSE /VERSION Input Category Customize Category Input Category Debug Category Input Category Customize Category Customize Category Customize Category Debug Category General Category Output Category Customize Category Output Category
Page 92 of 130
Besides discussing linker options individually, this section also discusses ModuleDefinition Files and Linker Reserved Words.
Compiler and Linker Options Generate Mapfile Enable Profiling Output Category Base Address Entry-Point Symbol Stack Allocations Version Information Input Category Object/Library Modules Ignore Libraries Ignore All Default Libraries Force Symbol References MS-DOS Stub File Name Customize Category Use Program Database Link Incrementally Program Database Name Output File Name Force File Output Print Progress Messages Suppress Startup Banner /MAP /PROFILE Command-Line Equivalent /BASE:address /ENTRY:function /STACK:reserve,commit /VERSION:major.minor Command-Line Equivalent filename on command line /NODEFAULTLIB:library /NODEFAULTLIB /INCLUDE:symbol /STUB:filename Command-Line Equivalent /PDB:filename /INCREMENTAL:{YES|NO} /PDB:filename /OUT:filename /FORCE /VERBOSE /NOLOGO
Page 93 of 130
Page 94 of 130
Debug Category Mapfile Name Generate Mapfile Generate Debug Info Microsoft Format COFF Format Both Formats
/ALIGN
Syntax: /ALIGN:number Specifies the alignment of each section within the linear address space of the program. The number argument is in bytes and must be a power of 2. The default is 4K. The linker generates a warning if the alignment produces an
Page 95 of 130
/BASE
Syntax: /BASE:{address | @filename,key} This option sets a base address for the program, overriding the default location for an executable file (at 0x400000) or a DLL (at 0x10000000). The operating system first attempts to load a program at its specified or default base address. If sufficient space is not available there, the system relocates the program. To prevent relocation, use the /FIXED option. Specify the preferred base address in the text box (or in the address argument on the command line). The linker rounds the specified number up to the nearest multiple of 64K. Another way to specify the base address is by using a filename, preceded by an at sign (@), and a key into the file. The filename is a text file that contains the locations and sizes of all DLLs your program uses. The linker looks for filename in either the specified path or, if no path is specified, in directories named in the LIB environment variable. Each line in filename represents one DLL and has the following syntax: key address size ;comment The key is a string of alphanumeric characters and is not case sensitive. It is usually the name of a DLL but it need not be. The key is followed by a base address in C-notation hexadecimal or decimal and a maximum size. All three arguments are separated by spaces or tabs. The linker issues a warning if the specified size is less than the virtual address space required by the program. Indicate a comment by a semicolon (;). Comments can be on the same or a separate line. The linker ignores all text from the semicolon to the end of the line. The following example shows part of such a file:
main one two 0x00010000 0x28000000 0x28100000 0x08000000 0x00100000 0x00300000 ; for PROJECT.EXE ; for DLLONE.DLL ; for DLLTWO.DLL
If the file that contains these lines is called DLLS.TXT, the following example command applies this information.
link dlltwo.obj /dll /base:dlls.txt,two
You can reduce paging and improve performance of your program by assigning base addresses so that DLLs do not overlap in the address space.
Page 96 of 130
An alternate way to set the base address is with the BASE argument in a NAME or LIBRARY module-definition statement. The /BASE and /DLL options together are equivalent to the LIBRARY statement. For information on module-definition statements, see Module-Definition Files.
/COMMENT
Syntax: /COMMENT:["] comment ["] Inserts a comment string into the header of an executable file or DLL, after the array of section headers. The type of operating system determines whether the string is loaded into memory. This comment string, unlike the comment specified with DESCRIPTION in a .DEF file, is not inserted into the data section. Comments are useful for embedding copyright and version information. To specify a comment that contains spaces or tabs, enclose it in double quotation marks ("). LINK removes the quotation marks before inserting the string. If more than one /COMMENT option is specified, LINK concatenates the strings and places a null byte at the end of each string.
/DEBUG
Syntax: /DEBUG Creates debugging information for the executable file or DLL. The linker puts the debugging information into a program database (PDB). It updates the program database during subsequent builds of the program. For details about PDBs, see /PDB. An executable file or DLL created for debugging contains the name and path of the corresponding PDB. Visual Fortran reads the embedded name and uses the PDB when you debug the program. The linker uses the base name of the program and the extension .PDB to name the PDB, and embeds the path where it was created. To override this default, use /PDB:filename. The object files must contain debugging information. Use the compiler's /Zi (Program Database), /Zd (Line Numbers Only), or /Z7 (C7 Compatible) option. If an object (whether specified explicitly or supplied from a library) was compiled with Program Database, its debugging information is stored in a PDB for the object file, and the name and location of the .PDB file is embedded in the object. The linker looks for the object's PDB first in the absolute path written in
Page 97 of 130
the object file and then in the directory that contains the object file. You cannot specify a PDB's filename or location to the linker. If you have turned off Use Program Database (or specified /PDB:NONE on the command line), or if you have chosen either /DEBUGTYPE:COFF or /DEBUGTYPE:BOTH, the linker does not create a PDB but instead puts the debugging information into the executable file or DLL. The /DEBUG option changes the default for the /OPT option from REF to NOREF.
/DEBUGTYPE
Syntax: /DEBUGTYPE:{CV|COFF|BOTH} This option generates debugging information in one of three ways, Microsoft format (CV), COFF format, or both:
l
/DEBUGTYPE:CV Visual Fortran requires new-style Microsoft Symbolic Debugging Information in order to read a program for debugging. To select this option in the Microsoft visual development environment, choose the Link tab of the Project Settings dialog box. In the Debug category, select the Microsoft Format button, which is only available if you have checked the Generate Debug Info box.
/DEBUGTYPE:COFF This option generates COFF-style debugging information. Some debuggers require Common Object File Format (COFF) debugging information. When you set this option, the linker does not create a PDB; in addition, incremental linking is disabled. To select this option in the visual development environment, choose the Link tab of the Project Settings dialog box. In the Debug category, select the COFF Format button, which is only available if you have checked the Generate Debug Info box.
/DEBUGTYPE:BOTH This option generates both COFF debugging information and old-style Microsoft debugging information. When you set this option, the linker does not create a PDB; in addition,
Page 98 of 130
incremental linking is disabled. The linker must call the CVPACK.EXE tool to process the old-style Microsoft debugging information. CVPACK must be in the same directory as LINK or in a directory in the PATH environment variable. In the visual development environment, specify this option with the Both Formats button, which is only available if you have selected Generate Debug Info. If you do not specify /DEBUG, /DEBUGTYPE is ignored. If you specify /DEBUG but not /DEBUGTYPE, the default type is /DEBUGTYPE:CV.
/DEF
Syntax: /DEF:filename Passes a module-definition (.DEF) file to the linker. Only one .DEF file can be specified to LINK. For details about .DEF files, see Module-Definition Files. When a .DEF file is used in a build, whether the main output file is an executable file or a DLL, LINK creates an import library (.LIB) and an exports file (.EXP). These files are created regardless of whether the main output file contains exports. Do not specify this option in the visual development environment; this option is for use only on the command line. To specify a .DEF file, add it to the project along with other files.
/DEFAULTLIB
Syntax: /DEFAULTLIB:libraries... Adds one or more libraries to the list of libraries that LINK searches when resolving references. A library specified with /DEFAULTLIB is searched after libraries specified on the command line and before default libraries named in object files. To specify multiple libraries, type a comma (,) between library names. Ignore All Default Libraries (/NODEFAULTLIB) overrides /DEFAULTLIB:library. Ignore Libraries (/NODEFAULTLIB:library) overrides /DEFAULTLIB:library when the same library name is specified in both.
Page 99 of 130
/DELAY
Syntax: /DELAY:{unload|nobind} Controls delayed loading of DDLs:
l
/DELAY:unload tells the delay-load helper function to support explicit unloading of the DLL by resetting the IAT to its original form, invalidating IAT pointers and causing them to be overwritten. /DELAY:nobind tells the linker not to include a bindable IAT in the final image. The resulting image cannot be statically bound. (Images with bindable IATs may be statically bound prior to execution.)
/DELAYLOAD
Syntax: /DELAYLOAD:dllname Causes delayed loading of DLLs. The dllname specifies a DLL to delay load. You can use this option as many times as necessary to specify as many DLLs as you choose. You must link your program with Delayimp.lib or implement your own delay-load helper function.
/DLL
Syntax: /DLL Builds a DLL as the main output file. A DLL usually contains exports that can be used by another program. There are three methods for specifying exports, listed in recommended order of use:
l l l
cDEC$ ATTRIBUTES DLLEXPORT in the source code An /EXPORT specification in a LINK command An EXPORTS statement in a module definition (.DEF) file
An alternate way to build a DLL is with the LIBRARY module-definition statement. The /BASE and /DLL options together are equivalent to the LIBRARY statement. In the visual development environment, you can set this option by choosing Dynamic-Link Library under Project Type in the New Project dialog box.
/DRIVER
Syntax: /DRIVER[:UPONLY] Use this option to build a Windows NT 4 or Windows 2000 kernel mode driver. The linker will perform some special optimizations if this option is selected. The UPONLY keyword causes the linker to add the IMAGE_FILE_UP_SYSTEM_ONLY bit to the characteristics in the output header to specify that it is a uniprocessor (UP) driver. The operating system will refuse to load a UP driver on a multiprocessor (MP) system.
/ENTRY
Syntax: /ENTRY:function This option sets the starting address for an executable file or DLL. Specify a function name that is defined with cDEC$ ATTRIBUTES STDCALL. The parameters and return value must be defined as documented in the Win32 API for WinMain (for an .EXE) or DllEntryPoint (for a DLL). It is recommended that you let the linker set the entry point. By default, the starting address is a function name from the run-time library. The linker selects it according to the attributes of the program, as shown in the following table. Function Name mainCRTStartup WinMainCRTStartup Default for An application using /SUBSYSTEM:CONSOLE; calls main An application using /SUBSYSTEM:WINDOWS; calls WinMain, which must be defined with the cDEC$ ATTRIBUTES STDCALL attribute
A DLL; calls DllMain (which must be defined with the STDCALL attribute) if it exists
If the /DLL or /SUBSYSTEM option is not specified, the linker selects a subsystem and entry point depending on whether main or WinMain is defined. The functions main, WinMain, and DllMain are the three forms of the userdefined entry point.
/EXETYPE
Syntax: /EXETYPE:DYNAMIC Used when building a virtual device driver (VxD). A VxD is linked using the /VXD option. Specify DYNAMIC to create a dynamically-loaded VxD.
/EXPORT
Syntax: /EXPORT:entryname[=internalname][, @ordinal [, NONAME]] [, DATA] Lets you export a function from your program to allow other programs to call the function. You can also export data. Exports are usually defined in a DLL. The entryname is the name of the function or data item as it is to be used by the calling program. You can optionally specify the internalname as the function known in the defining program; by default, internalname is the same as entryname. The ordinal specifies an index into the exports table in the range 1 65535; if you do not specify ordinal, LINK assigns one. The NONAME keyword exports the function only as an ordinal, without an entryname. The DATA keyword specifies that the exported item is a data item. The data item in the client program must be declared using DLLIMPORT. (The CONSTANT keyword is supported for compatibility but is not recommended.) There are three methods for exporting a definition, listed in recommended order of use:
l l l
cDEC$ ATTRIBUTES DLLEXPORT in the source code An /EXPORT specification in a LINK command An EXPORTS statement in a module definition (.DEF) file
All three methods can be used in the same program. When LINK builds a program that contains exports, it also creates an import library, unless an .EXP file is used in the build. LINK uses decorated forms of identifiers. A "decorated name" is an internal representation of a procedure name or variable name that contains information about where it is declared; for procedures, the information includes how it is called. Decorated names are mainly of interest in mixed-language programming, when calling Fortran routines from other languages. The compiler decorates an identifier when it creates the object file. If entryname or internalname is specified to the linker in its undecorated form as it appears in the source code, LINK attempts to match the name. If it cannot find a unique match, LINK issues an error. Use the DUMPBIN tool described in Examining Files with DUMPBIN to get the decorated form of an identifier when you need to specify it to the linker. Do not specify the decorated form of identifiers declared with the C or STDCALL attributes. For more information on when and how to use decorated names, see Adjusting Naming Conventions in Mixed-Language Programming.
/FIXED
Syntax: /FIXED This option tells the operating system to load the program only at its preferred base address. If the preferred base address is unavailable, the operating system does not load the file. For more information on base address, see /BASE. When you specify /FIXED, LINK does not generate a relocation section in the program. At run-time, if the operating system cannot load the program at that address, it issues an error and does not load the program. Some Win32 operating systems, especially those that coexist with MS-DOS, frequently must relocate a program. A program created with /FIXED will not run on Win32s operating systems. Note: Do not use /FIXED when building device drivers.
/FORCE
Syntax: /FORCE:[ {MULTIPLE|UNRESOLVED} ]
Tells the linker to create a valid executable file or DLL even if a symbol is referenced but not defined or is multiply defined. The /FORCE option can take an optional argument:
l
Use /FORCE:MULTIPLE to create an output file whether or not LINK finds more than one definition for a symbol. Use /FORCE:UNRESOLVED to create an output file whether or not LINK finds an undefined symbol.
A file created with this option may not run as expected. The linker will not link incrementally with the /FORCE option. You can select this option in the visual development environment by checking the Force File Output box in the Customize category of the Link tab in the Project Settings dialog box.
/HEAP
Syntax: /HEAP:reserve[,commit] Sets the size of the heap in bytes. The reserve argument specifies the total heap allocation in virtual memory. The default heap size is 1MB. The linker rounds up the specified value to the nearest 4 bytes. The optional commit argument is subject to interpretation by the operating system. In Windows NT 4 and Windows 2000, it specifies the amount of physical memory to allocate at a time. Committed virtual memory causes space to be reserved in the paging file. A higher commit value saves time when the application needs more heap space but increases the memory requirements and possibly startup time. Specify the reserve and commit values in decimal or C-language notation. (Use the digits 1 - 9 for decimal values, precede octal values with zero (0), and precede hexadecimal values with zero and x (0x or 0X).
/IMPLIB
Syntax: /IMPLIB:filename
Overrides the default name for the import library that LINK creates when it builds a program that contains exports. The default name is formed from the base name of the main output file and the extension .LIB. A program contains exports if one or more of the following is true:
l l l
cDEC$ ATTRIBUTES DLLEXPORT in the source code An /EXPORT specification in a LINK command An EXPORTS statement in a module definition (.DEF) file
LINK ignores the /IMPLIB option when an import library is not being created. If no exports are specified, LINK does not create an import library. If an export (.EXP) file is used in the build, LINK assumes an import library already exists and does not create one. For information on import libraries and export files, see Import Libraries and Export Files in Using Visual Fortran Tools.
/IMPORT
Syntax: /IMPORT This option is specific to MACOS and does not apply to Visual Fortran.
/INCLUDE
Syntax: /INCLUDE:symbol Tells the linker to add a specified symbol to the symbol table. Specify a symbol name in the text box. To specify multiple symbols, specify /INCLUDE:symbol once for each symbol. The linker resolves symbol by adding the object that contains the symbol definition to the program. This is useful for including a library object that otherwise would not be linked to the program. Specifying a symbol in the /INCLUDE option overrides the removal of that symbol by /OPT:REF. To select this option in the visual development environment, choose the Force Symbol References text box in the Input category of the Link tab of the Project Settings dialog box.
/INCREMENTAL
Syntax: /INCREMENTAL:{YES|NO} Controls how the linker handles incremental linking. By default, the linker runs in incremental mode (for exceptions, see the /[no]traceback compiler option). To prevent incremental linking, clear the Link Incrementally check box in the Customize category (or specify /INCREMENTAL:NO on the command line). To link incrementally, set the Link Incrementally check box (or specify /INCREMENTAL:YES on the command line). When you specify this option, the linker issues a warning if it cannot link incrementally and then links the program nonincrementally. Certain options and situations override /INCREMENTAL:YES. Most programs can be linked incrementally. However, some changes are too great, and some options are incompatible with incremental linking. LINK performs a full link if any of the following options are specified:
l l l l l l l
Link Incrementally is turned off (/INCREMENTAL:NO) COFF Format (/DEBUGTYPE:COFF) Both Formats (/DEBUGTYPE:BOTH) /OPT:REF is selected. /OPT:ICF is selected. /ORDER is selected. Use Program Database is not selected (/PDB:NONE) when Generate Debug Info (/DEBUG) is selected.
l l l
l l
The incremental status (.ILK) file is missing. (LINK creates a new .ILK file in preparation for subsequent incremental linking.) There is no write permission for the .ILK file. (LINK ignores the .ILK file and links nonincrementally.) The .EXE or .DLL output file is missing. The timestamp of the .ILK, .EXE, or .DLL is changed. A LINK option is changed. Most LINK options, when changed between builds, cause a full link. An object (.OBJ) file is added or omitted. An object that was compiled with the /Yu /Z7 option is changed.
To select this option in the visual development environment, select the Link Incrementally check box in the Customize category of the Link tab in the Project
/LARGEADDRESSAWARE
Syntax: /LARGEADDRESSAWARE Informs the linker that the application can handle addresses larger than 2 gigabytes.
/LIBPATH
Syntax: /LIBPATH:dir Overrides the environment library path. The linker will first search in the path specified by this option, and then search in the path specified in the LIB environment variable. You can specify only one directory for each /LIBPATH option you enter. If you want to specify more than one directory, you must specify multiple /LIBPATH options. The linker will then search the specified directories in order.
/LINK50COMPAT
Syntax: /LINK50COMPAT Generates import libraries in the old (Visual C++ version 5.0) format for backward compatibility.
/MACHINE
Syntax: /MACHINE:{IX86|IA64} Specifies the target platform for the program (for Visual Fortran, specify either IX86 or IA64). Usually, you do not need to specify the /MACHINE option. LINK infers the machine type from the .OBJ files. However, in some circumstances LINK cannot determine the machine type and issues a linker tools error LNK1113. If such an error occurs, specify /MACHINE.
/MAP
Syntax: /MAP[:filename] This option tells the linker to generate a mapfile. You can optionally specify a map file name to override the default. The linker names the mapfile with the base name of the program and the extension .MAP. To override the default name, use the filename argument. A map file is a text file that contains the following information about the program being linked:
l l l
l l
The module name, which is the base name of the file The timestamp from the program file header (not from the file system) A list of groups in the program, with each group's start address (as section:offset), length, group name, and class A list of public symbols, with each address (as section:offset), symbol name, flat address, and object file where the symbol is defined The entry point (as section:offset) A list of fixups
To select this in the visual development environment, select the Generate Mapfile check box in the Debug category of the Link tab in the Project Settings dialog box.
/MAPINFO
Syntax: /MAPINFO:{EXPORTS|FIXUPS|LINES} Informs the linker to include the specified information in a map file, which is created if you specify the /MAP option:
l l
EXPORTS tells the linker to include exported functions. FIXUPS tells the linker to include base-relocation information in the mapfile if relocation information exists in the image. Base relocations will be present in the image if you link with /FIXED:NO. LINES includes line-number information.
/MERGE
Syntax: /MERGE:from=to Combines the first section (from) with the second section (to), naming the resulting section to. If the second section does not exist, LINK renames the section from as to. The /MERGE option is useful for creating virtual device drivers (VxDs) and overriding the compiler-generated section names.
/NODEFAULTLIB
Syntax: /NODEFAULTLIB[:library] Tells the linker to remove all default libraries from the list of libraries it searches when resolving external references. If you specify library, the linker only ignores the libraries you have named. To specify multiple libraries, type a comma (,) between the library names. The linker resolves references to external definitions by searching first in libraries specified on the command line, then in default libraries specified with the /DEFAULTLIB option, then in default libraries named in object files. Ignore All Default Libraries (/NODEFAULTLIB) overrides /DEFAULTLIB:library. Ignore Libraries (/NODEFAULTLIB:library) overrides /DEFAULTLIB:library when the same library name is specified in both. To select this in the visual development environment, select the Ignore Libraries or Ignore All Default Libraries check box in the Input category of the Link tab in the Project Settings dialog box.
/NOENTRY
Syntax: /NOENTRY This option is required for creating a resource-only DLL. Use this option to prevent LINK from linking a reference to _main into the DLL.
/NOLOGO
Syntax: /NOLOGO The /nologo option prevents display of the copyright message and version number. This option also suppresses echoing of command files. For the command line and in the visual development environment for a debug configuration, the default is /logo (displays copyright message, version number, and echoing of command files). In the visual development environment for a release configuration, the default is /nologo. By default, information displayed by /logo is sent by the linker to the Output window. On the command line, it is sent to standard output and can be redirected to a file. To select this option in the visual development environment, select the Suppress Startup Banner check box in the Customize category of the Link tab in the Project Settings dialog box.
/OPT
Syntax: /OPT:{REF|NOREF} Controls the optimizations LINK performs during a build. Optimizations generally decrease the image size and increase the program speed, at a cost of increased link time. By default, LINK removes unreferenced packaged functions (COMDATs). This optimization is called transitive COMDAT elimination. To override this default and keep unused packaged functions in the program, specify /OPT:NOREF. You can use the /INCLUDE option to override the removal of a specific symbol. It is not possible to create packaged functions with the Visual Fortran compiler. This description is included for mixed-language applications with languages such as Visual C++ that support packaged functions (with the /Gy compiler option). If you specify the /DEBUG option, the default for /OPT changes from REF to NOREF and all functions are preserved in the image. To override this default and optimize a debugging build, specify /OPT:REF. The /OPT:REF option disables incremental linking.
/ORDER
Syntax:
/ORDER:@filename Lets you perform optimization by telling LINK to place certain packaged functions into the image in a predetermined order. It is not possible to make packaged functions with the Visual Fortran compiler. This description is included for mixed-language applications with languages such as Visual C++ that support packaged functions (with the /Gy compiler option). LINK places packaged functions in the specified order within each section in the image. Specify the order in filename, which is a text file that lists the packaged functions in the order you want to link them. Each line in filename contains the name of one packaged function. Function names are case sensitive. A comment is specified by a semicolon (;) and can be on the same or a separate line. LINK ignores all text from the semicolon to the end of the line. LINK uses decorated forms of identifiers. A decorated name is an internal representation of a procedure name or variable name that contains information about where it is declared; for procedures, the information includes how it is called. Decorated names are mainly of interest in mixed-language programming, when calling Fortran routines from other languages. The compiler decorates an identifier when it creates the object file. If the name of the packaged function is specified to the linker in its undecorated form as it appears in the source code, LINK attempts to match the name. If it cannot find a unique match, LINK issues an error. Use the DUMPBIN tool to get the decorated form of an identifier when you need to specify it to the linker. Do not specify the decorated form of identifiers declared with cDEC$ ATTRIBUTES C or STDCALL. For more information on when and how to use decorated names, see Adjusting Naming Conventions in MixedLanguage Programming. If more than one /ORDER specification is used, the last one specified takes effect. Ordering allows you to optimize your program's paging behavior through swap tuning. Group a function with the functions it calls. You can also group frequently called functions together. These techniques increase the probability that a called function is in memory when it is needed and will not have to be paged from disk. This option disables incremental linking.
/OUT
Syntax: /OUT:filename Overrides the default name and location of the image file that LINK creates. By default, LINK forms the filename using the base name of the first file specified and the appropriate extension (.EXE or .DLL). The /OUT option controls the default base name for a mapfile or import library. For details, see the descriptions of /MAP and /IMPLIB.
/PDB
Syntax: /PDB[:filename] Controls how the linker produces debugging information. The optional filenameargument overrides the default filename for the program database. The default filename for the PDB has the base name of the program and the extension .PDB. By default when you specify /DEBUG, the linker creates a program database (PDB), which holds debugging information. If you have not specified /DEBUG, the linker ignores /PDB. If you specify /PDB:NONE, the linker does not create a PDB, but instead puts old-style debugging information into the executable file or DLL. The linker then calls the CVPACK.EXE tool, which must be in the same directory as LINK.EXE or in a directory in the PATH environment variable. Debugging information in a program database must be in Microsoft Format (/DEBUGTYPE:CV). If you choose either COFF Format (/DEBUGTYPE:COFF) or Both Formats (/DEBUGTYPE:BOTH), no PDB is created. Incremental linking is suppressed if you specify /PDB:NONE. You can select this option in the visual development environment by selecting the Use Program Database check box in the Customize category of the Link tab in the Project Settings dialog box.
/PDBTYPE
Syntax: /PDBTYPE:{CON[SOLIDATE]|SEPT[YPES]}
Controls which Program Database (PDB) stores the debug type information. On the command line, the /PDBTYPE option can take one of the following arguments:
l
Use /PDBTYPE:CON[SOLIDATE] to tell the linker to place the debug type information in a single .PDB file. This option is the default. This option cannot be used if /PDB:NONE is specified. Use /PDBTYPE:SEPT[YPES] to tell the linker to leave the debug type information distributed in the source (compiler) .PDB files. In the Project Settings dialog box, select Separate Types in the Debug category of the Link tab to specify this linker option.
The debugger startup time may be slightly faster. Less hard disk space is needed to store data.
The disadvantage is more files are needed for the debugger to find the debug information when debugging. Use this option when you plan to debug privately. Do not use this option when the debug build is to be used by multiple users.
/PROFILE
Syntax: /PROFILE Creates an output file that can be used with the profiler. This option is found only in the General category on the Link tab. A profiler-ready program has a map file. If it contains debugging information, the information must be stored in the output file instead of a program database file (.PDB file) and must be in Microsoft old-style format. In the visual development environment, setting Enable Profiling enables the Generate Mapfile option in the General and Debug categories. If you set the Generate Debug option, be sure to choose Microsoft Format in the Debug category. On the command line, /PROFILE has the same effect as setting the /MAP option; if the /DEBUG option is specified, then /PROFILE also implies the options /DEBUGTYPE:CV and /PDB:NONE. In either case, /PROFILE
You can select this option in the visual development environment by selecting the Enable Profiling check box in the General category of the Link tab in the Project Settings dialog box.
/RELEASE
Syntax: /RELEASE Sets the checksum in the header of an executable file. The operating system requires the checksum for certain files such as device drivers. To ensure compatibility with future operating systems, set the checksum for release versions of your programs. This option is set by default when you specify the /SUBSYSTEM:NATIVE option.
/SECTION
Syntax: /SECTION:name,[E][C][I][R][W][S][D][K][L][P][X] Changes the properties of a section, overriding the properties set when the .OBJ file for the section was compiled. A section in a portable executable (PE) file is roughly equivalent to a segment or the resources in an NE file. Sections contain either code or data. Unlike segments, sections are blocks of contiguous memory with no size constraints. Some sections contain code or data that your program declared and uses directly, while other data sections are created for you by the linker and librarian, and contain information vital to the operating system. Specify a colon (:) and a section name. The name is case sensitive. Specify one or more properties for the section. The property characters, listed below, are not case sensitive. You must specify all properties that you want the section to have; an omitted property character causes that property bit to be turned off. The meanings of the property characters are shown below.
Character E C I R W S D K L P X
Property Execute Conforming IOPL Read Write Shared Discardable Cacheable Preload Pageable Memoryresident
Meaning Allows code to be executed Marks the section as conforming Marks the section as IOPL Allows read operations on data Allows write operations on data Shares the section among all processes that load the image Marks the section as discardable Marks the section as not cacheable VxD only; marks the section as preload Marks the section as not pageable VxD only; marks the section as memoryresident
/STACK
Syntax: /STACK:reserve[,commit] Sets the size of the stack in bytes. The reserve argument specifies the total stack allocation in virtual memory. The default stack size is 1MB. The linker rounds up the specified value to the nearest 4 bytes. The optional commit argument is subject to interpretation by the operating system. In Windows NT 4 and Windows 2000, it specifies the amount of physical memory to allocate at a time. Committed virtual memory causes space to be reserved in the paging file. A higher commit value saves time when the application needs more stack space but increases the memory requirements and possibly startup time.
Specify the reserve and commit values in decimal or C-language notation. (Use the digits 1-9 for decimal values, precede octal values with zero (0), and precede hexadecimal values with zero and x (0x or 0X). An alternate way to set the stack is with the STACKSIZE statement in a .DEF file. STACKSIZE overrides Stack Allocations (/STACK) if you specify both. You can change the stack after the executable file is built by using the EDITBIN.EXE tool. For more information, see Editing Files with EDITBIN. To set these options in the visual development environment, type values in the Reserve and Commit boxes in the Output category of the Link tab in the Project Settings dialog box.
/STUB
Syntax: /STUB:filename Attaches an MS-DOS stub program to a Win32 program. A stub program is invoked if the file is executed in MS-DOS. Usually, it displays an appropriate message; however, any valid MS-DOS application can be a stub program. Specify a filename for the stub program after a colon (:). The linker checks filename to be sure that it is a valid MS-DOS executable file and issues an error if the file is not valid. The program must be an .EXE file; a .COM file is invalid for a stub program. If you do not specify /STUB, the linker attaches a default stub program that generates the following message:
This program cannot be run in MS-DOS mode.
You can select this option in the visual development environment by typing the stub file name in the MS-DOS Stub File Name box in the Input category of the Link tab of the Project Settings dialog box.
/SUBSYSTEM
Syntax: /SUBSYSTEM:{CONSOLE|WINDOWS|NATIVE|POSIX|WINDOWSCE} [,major [.minor]]
Tells the operating system how to run the executable file. The subsystem is specified as follows:
l
l l
The CONSOLE subsystem is used for Win32 character-mode applications. Console applications are given a console by the operating system. If main or wmain is defined, CONSOLE is the default. The WINDOWS subsystem is appropriate for an application that does not require a console. It creates its own windows for interaction with the user. If WinMain or wWinMain is defined, WINDOWS is the default. The NATIVE subsystem is used for device drivers. The POSIX subsystem creates an application that runs with the POSIX subsystem in Windows NT 4 or Windows 2000. The WINDOWSCE subsystem is not used by Visual Fortran.
The optional major and minor version numbers specify the minimum required version of the subsystem. The arguments are decimal numbers in the range 0 65535. The default is version 3.10 for CONSOLE and WINDOWS and 1.0 for NATIVE. The choice of subsystem affects the default starting address for the program. For more information, see the /ENTRY option.
/SWAPRUN
Syntax: /SWAPRUN:{NET|CD} Tells the operating system to first copy the linker output to a swap file, and then run the image from there. This is a Windows NT 4 or Windows 2000 feature. If NET is specified, the operating system will first copy the binary image from the network to a swap file and load it from there. This option is useful for running applications over the network. When CD is specified, the operating system will copy the image on a removable disk to a page file and then load it.
/VERBOSE
Syntax: /VERBOSE[:LIB] The linker sends information about the progress of the linking session to the Output window. If specified on the command line, the information is sent to standard output and can be redirected to a file.
The displayed information includes the library search process and lists each library and object name (with full path), the symbol being resolved from the library, and the list of objects that reference the symbol. Adding :LIB to the /VERBOSE option restricts progress messages to those indicating the libraries searched. You can select this option in the Microsoft visual development environment by filling in the Print Progress Messages box in the Customize category of the Link tab of the Project Settings dialog box.
/VERSION
Syntax: /VERSION:major [.minor ] Tells the linker to put a version number in the header of the executable file or DLL. The major and minor arguments are decimal numbers in the range 0 - 65535. The default is version 0.0. An alternate way to insert a version number is with the VERSION moduledefinition statement. You can select this option in the Microsoft visual development environment by typing version information in the Major and Minor boxes in the Output category of the Link tab of the Project Settings dialog box.
/VXD
Syntax: /VXD Creates a virtual device driver (VxD). When this option is specified, the default file name extension changes to .VXD. For details on VxDs, see the Microsoft Windows NT Device Driver Kit. A .VXD file is not in Common Object File Format, and it cannot be used with DUMPBIN or EDITBIN. It does not contain debugging information. However, you can create a map file when you link a .VXD file. A .VXD file cannot be incrementally linked.
/WARN
Syntax: /WARN[:level] Allows you to determine the output of LINK warnings. Specify the level as one of the following: level 0 1 Meaning Suppress all warnings. Displays most warnings. Overrides a /WARN:level specified earlier on the LINK command line or in the LINK environment variable. Default if /WARN:level is not used. Displays additional warnings. Default if /WARN is specified without level.
/WS
Syntax: /WS:AGGRESSIVE Adds the WS_AGGRESSIVE property to your application's image. The Windows NT 4 or Windows 2000 loader will recognize this property and aggressively trim the working set of the process when it is not active. Using this option is similar to adding the following call throughout your application:
SetProcessWorkingSetSize(hThisProcess, -1, -1)
/WS:AGGRESSIVE can be used for applications that must have a low impact on the system's memory pool. If the speed of your application is important, do not use /WS:AGGRESSIVE without testing the resulting performance implications. Ideal candidates are processes that tend to operate in the background, such as services and screen savers.
Module-Definition Files
A module-definition (.DEF) file is a text file that contains statements that define an executable file or DLL. (These should not be confused with module program units, described in Program Units and Procedures.) The following sections describe the statements in a .DEF file and Rules for Module-Definition Statements discusses rules for DEF files. Because LINK provides equivalent command-line options for most moduledefinition statements, a typical program for Win32 does not usually require a .DEF file. In contrast, 16-bit programs for Windows almost always must be linked using a .DEF file. You can use one or more of the following statements in a .DEF file:
l l l l l l
The section describing each module-definition statement gives its command-line equivalent.
Statements and attribute keywords are not case sensitive. User-specified identifiers are case sensitive. Use one or more spaces, tabs, or newline characters to separate a statement keyword from its arguments and to separate statements from each other. A colon (:) or equal sign (=) that designates an argument is surrounded by zero or more spaces, tabs, or newline characters. A NAME or LIBRARY statement, if used, must precede all other statements. Most statements appear only once in the .DEF file and accept one specification of arguments. The arguments follow the statement keyword on the same or subsequent line(s). If the statement is repeated with different arguments later in the file, the latter statement overrides the former. The EXPORTS statement can appear more than once in the .DEF file. Each statement can take multiple specifications, which must be separated by one
or more spaces, tabs, or newline characters. The statement keyword must appear once before the first specification and can be repeated before each additional specification.
l
Comments in the .DEF file are designated by a semicolon (;) at the beginning of each comment line. A comment cannot share a line with a statement, but it can appear between specifications in a multiline statement. (EXPORTS is a multiline statement.) Numeric arguments are specified in decimal or in C-language notation. If a string argument matches a reserved word, it must be enclosed in double quotation (" ) marks.
Many statements have an equivalent LINK command-line option. See the Linker Options and Related Information for additional details.
DESCRIPTION
Syntax: DESCRIPTION "text" This statement writes a string into an .rdata section. Enclose the specified text in single or double quotation marks (' or "). To use a literal quotation mark (either single or double) in the string, enclose the string with the other type of mark. This feature differs from the comment specified with the /COMMENT linker option.
EXPORTS
Syntax: EXPORTS This statement makes one or more definitions available as exports to other programs. EXPORTS marks the beginning of a list of export definitions. Each definition must be on a separate line. The EXPORTS keyword can be on the same line as the first definition or on a preceding line. The .DEF file can contain one or more EXPORTS statements. The syntax for an export definition is:
entryname[=internalname] [@ordinal [ NONAME]] [DATA] For information on the entryname, internalname, ordinal, NONAME, and DATA arguments, see the /EXPORT option. There are three methods for exporting a definition, listed in recommended order of use:
l l l
cDEC$ ATTRIBUTES DLLEXPORT in the source code An /EXPORT specification in a LINK command An EXPORTS statement in a .DEF file
All three methods can be used in the same program. When LINK builds a program that contains exports, it also creates an import library, unless the build uses an .EXP file.
LIBRARY
Syntax: LIBRARY [library] [BASE=address] This statement tells LINK to create a DLL. LINK creates an import library at the same time, unless you use an .EXP file in the build. The library argument specifies the internal name of the DLL. (Use the Output File Name (/OUT) option to specify the DLL's output name.) The BASE=address argument sets the base address that the operating system uses to load the DLL. This argument overrides the default DLL location of 0x10000000. See the description of the Base Address (/BASE) option for details about base addresses. You can also use the /DLL linker option to specify a DLL build, and the /BASE option to set the base address.
NAME
Syntax: NAME [application] [BASE=address] This statement specifies a name for the main output file. An equivalent way to specify an output filename is with the /OUT option, and an equivalent way to set the base address is with the /BASE option. If both are specified, /OUT overrides NAME. See the Base Address (/BASE) and Output File Name (/OUT) options for
Compiler and Linker Options details about output filenames and base addresses.
STACKSIZE
Syntax: STACKSIZE reserve [,commit] This statement sets the size of the stack in bytes. An equivalent way to set the stack is with the /STACK option. See the /STACK option for details about the reserve and commit arguments.
VERSION
Syntax: VERSION major [.minor] This statement tells LINK to put a number in the header of the executable file or DLL. The major and minor arguments are decimal numbers in the range 0 65535. The default is version 0.0. An equivalent way to specify a version number is with the Version Information (/VERSION) option.
Compiler and Linker Options DISCARDABLE DYNAMIC EXECUTE-ONLY EXECUTEONLY EXECUTEREAD EXETYPE EXPORTS FIXED FUNCTIONS HEAPSIZE IMPORTS IMPURE INCLUDE MULTIPLE NAME NEWFILES NODATA NOIOPL NONAME NONCONFORMING NONDISCARDABLE NONE NONSHARED NOTWINDOWCOMPAT OBJECTS OLD RESIDENTNAME SEGMENTS SHARED SINGLE STACKSIZE STUB VERSION WINDOWAPI WINDOWCOMPAT WINDOWS
Using the DF or FL32 Command Line Equivalent Visual Fortran Compiler Options
FL32 requests no optimization (/Od). See /[no]optimize. FL32 requests checking of arguments passed to and results from the math library (/math_library:check or /Od). See /[no]math_library. FL32 provides minimal debug information (/debug:minimal or /Zd). See /[no]debug. FL32 requests full Microsoft Fortran PowerStation compatibility
l l
(/fpscomp:all). See /[no]fpscomp. FL32 disallows alternative PARAMETER syntax (/noaltparam). See /[no] altparam. FL32 requests record length units for unformatted files to be in bytes (/assume:byterecl). See /assume. FL32 requests warnings for mismatched arguments (/warn:argument_checking). See /[no]warn. FL32 compiles each source unit individually and retains intermediate files that would otherwise be deleted (/keep). This prevents interprocedure optimizations at higher optimization levels. See /keep. FL32 does not display an informational message related to compiling multiple files individually. See /warn:fileopts. FL32 requests no inlining (/inline:none). See /[no]inline. FL32 places module files in the same directory as the object files. See /module:path.
Recognize the same set of command-line options. For example, the following commands are supported:
DF /Odx test2.for FL32 /Odx test2.for
Both DF and FL32 command lines allow most Microsoft Fortran PowerStation style options (such as /Ox) and all Visual Fortran options (such as /optimize:4). For a detailed list of equivalent Microsoft Fortran PowerStation style compiler options and Visual Fortran compiler options, see Equivalent Visual Fortran Compiler Options.
l
Activate the same compiler, the Compaq Fortran compiler. For new programs and most existing applications, use the Compaq Fortran compiler (default). The Compaq Fortran compiler and language used by Visual Fortran provides a superset of the Fortran 95 standard with extensions for compatibility with previous versions of Compaq Fortran (DIGITAL Fortran, DEC Fortran), VAX FORTRAN, and Microsoft Fortran PowerStation Version 4.
Pass options specified after /LINK to the LINK command. The LINK command options after /link are passed directly to the Linker. These options are described in Linker Options.
Allow the use of indirect command files. For example, assume the file text.txt contains the following:
Either of the following (DF or FL32) commands executes the contents of file text.txt as an indirect command file to create a debugging version of the executable program and its associated PDB file:
DF @test.txt
FL32 @test.txt
To request Microsoft Fortran PowerStation V4 compatibility, specify the /[no] fpscomp option. For information about using the DF command, see Using the Compiler and Linker from the Command Line.
/FAs
/FAcs
Page 126 of 130 Assembly listing to file file. Specify /asmfile[:file] with /noasmattributes or specify /Fa[file]. Assembly listing with source and machine code to file file. Specify /asmfile[:file] with /asmattributes:all or specify /Fc[file]. Assembly listing with machine instructions to file file. Specify /asmfile[:file] with /asmattributes:machine or specify /Fl[file]. Source listing with compiled code. Specify /list [:file] with /show:map or specify /Fs[file].
/Fl[file]
Generates extended Source Browser information. Specify /browser[:file] or /FR[file]. Generates code for specific ia32 chip architectures. Specify /tune:keyword. Automatic inlining of code, use with /Ox. Specify /inline:speed or /Ob2. No code optimization (default for FL32 command). Specify /optimize:0 with /math_library:check, or specify /Od. Improved floating-point consistency. Specify /fltconsistency or /Op. Applies to ia32 systems only. Full optimization with no error checking. Specify /optimize:4 with /math_library:fast and /assume:nodummy_aliases, or specify /Ox. Speed optimization and denoted inlining; error checking. Specify /optimize:4 with /assume:nodummy_aliases and (on ia32 systems) /math_library:check with /fltconsistency, or specify /Oxp. Packs structures on n-byte boundary (n is 1, 2, or 4). Specify /alignment[:keyword] or /Zp[n]. If you specify /Zp (omit n), structures are packed at 8-byte boundaries.
/Op
/Ox
/Oxp
/Zp[n]
Line length for Fortran 90 fixed-form source (nn is 72, 80, or 132). Specify /extend_source[:nn] or /4Lnn. Enable/disable extended error checking. Specify /check[:keyword], /4Yb, or /4Nb. Warnings about undeclared variables. Specify /warn: [no]declarations, /4Yd or /4Nd. Suppress warnings. Specify /nowarn or /W0. Show warnings (default). Specify /warn:general or /W1. Interpret all warnings as errors. Specify /warn: (general,errors) or /WX.
Language Standard, Source Form, and Data Options /4Ya or /4Na /4Yaltparam /4Naltparam Makes all variables AUTOMATIC. Specify /[no] automatic, /[no]static, /4Ya, or /4Na. Use the alternate syntax for PARAMETER statements. Specify /[no]altparam, /4Yaltparam, or /4Naltparam. Use free-form source format. Specify /[no]free, /[no]fixed, /4Yf, or /4Nf. Change default KIND for INTEGER and LOGICAL declarations. Specify /integer_size:nn (nn is 16 for KIND=2) or /4I2. Change default KIND for REAL declarations. Specify /real_size:nn (nn is 32 for KIND=4) or /4R8. Strict Fortran 90 syntax. Specify /stand:f90, /warn:stderrors, /4Ys, or /4Ns.
/4R8
/4Ys or /4Ns
Compiler Directive Options /Dsymbol[=int] Define preprocessor symbol. Specify /define:symbol[=int] or Dsymbol[=int].
Page 128 of 130 Treat lines with d or D in column 1 as comments. Specify /d_lines or /4ccd or /4ccD (partial support).
Build Control Options /4Yportlib or /4Nportlib /Fd[file] /Fe[file] /Fm[file] /Fo[file] /GNa Specify /4Yportlib or /4Nportlib. Controls creation of compiler PDB files. Specify /[no]pdbfile[:ile] or /Fd[file]. f Specifies file name of executable or DLL file. Specify /exe:file, /dll:file, or /Fe[file]. Controls creation of link map file. Specify /map [:file] or /Fm[file]. Controls creation of object file. Specify /object [:file] or /Fo[file]. Keep external names as is and treat source code identifiers as case sensitive. Specify /names:as_is or /GNa. Make external names lowercase and ignore the case of source code identifiers. Specify /names:lowercase or /GNl. Make external names uppercase and ignore the case of source code identifiers. Specify /names:uppercase or /GNu. Control search path for module or include files. Specify /[no]include[:path] or /Ipath. Create dynamic-link library. Specify /dll or /LD. Link against multithreaded DLL libraries. Specify /libs:dll with /threads or /MD. Link against multithreaded DLL libraries. Specify /libs:dll with /threads and /dbglibs or specify /MDd. Link against single threaded DLL libraries. Specify /libs:dll or /MDs.
/GNl
/GNu
/MDs
Page 129 of 130 Link against libraries for windows applications. Specify /winapp or /MG. Link against single threaded static libraries. Specify /libs:static or /ML. Link against single threaded static libraries. Specify /libs:static with /dbglibs or /MLd. Link against multithreaded static libraries. Specify /libs:static with /threads or /MT. Link against multithreaded static libraries. Specify /libs:static with /threads and /dbglibs or specify /MTd. Link against QuickWin multidoc libraries. Specify /libs:qwin or /MW. Link against QuickWin single doc libraries. Specify /libs:qwins or /MWs. Request that file be treated as a Fortran source file. Specify /source:filename or /Tffile. Place string in object file. Specify /bintext:string or /V"string". Request full debug information in object file. Specify /debug:full with /nopdbfile or /Z7. Request minimal debug information. Specify /debug:minimal with /pdbfile or /Zd. Request full debug information and create PDB file. Specify /debug:full with /pdbfile or /Zi. Do not insert any library names in object file. Specify /nolibdir or /Zla. Do not insert default library names in object file. Specify /libdir:noautomatic or /Zl. Perform syntax check only (no object). Specify /syntax_only or /Zs.
/MW /MWs /Tffile /V"string" /Z7 /Zd /Zi /Zla /Zl /Zs
Page 130 of 130 Requests traceback information (run-time program counter to source file line correlation). Specify /traceback or /Zt. Begin specifying linker options. Specify /link [option].
/link [option]
Command-Line Specific Options /?, /help /nologo Display command help. Specify /? or /help. Prevent display of copyright information. Specify /nologo.
Page 1 of 28
Preparing Your Program for Debugging Debugging the Squares Example Program Using Breakpoints in the Debugger Viewing Fortran Data Types in the Debugger Using the Array Viewer in the Debugger Locating Run-Time Errors in the Debugger
When developing your application with the Microsoft visual development environment When developing your application with the command-line environment
To prepare your program for debugging when using the visual development environment: 1. Start the visual development environment (click Developer Studio in the Compaq Visual Fortran program folder). 2. Open the appropriate Workspace (File menu, either Open Workspaces or Recent Workspaces). 3. Click the FileView pane. 4. To edit the source file to be debugged, double-click on the file name. 5. Click the Project name. The screen might appear as follows (the ClassView tab only appears if Visual C++ is also installed):
Page 2 of 28
6. In the Build menu, click Set Active Configuration and select the debug configuration. 7. To check your project settings for compiling and linking, in the Project menu, click Settings, then click the Fortran tab. Similarly, to check the debug options set for your program (such as program arguments or working directory), click the Debug tab in the Project Settings dialog box. 8. To compile your program: Click (select) the source file to be compiled In the Build menu, click Compile filename 9. Eliminate any compiler diagnostic messages in the text editor and recompile if needed. 10. To build your application, in the Build menu, click Build file.EXE. 11. Set breakpoints in the source file and debug the program, as described in Debugging the Squares Example Program. To prepare your program for debugging when using the command line (DF command):
Page 3 of 28
1. Correct any compilation and linker errors. 2. In a command window (such as the Fortran command window available from the Visual Fortran program folder), compile and link the program with full debug information and no optimization:
DF /debug:full /nooptimize file.f90
3. Start the visual development environment. 4. In the File menu, click the Open Workspace item. Specify the file name of the executable (.EXE) file to be debugged. 5. In the File menu, click Open. Specify the name of the source file (such as .F90 or .FOR) that corresponds to the file being debugged. The text editor window appears. Your screen might appear as follows:
6. Set breakpoints in the source file and debug the program, as described in Debugging the Squares Example Program. To add Source Browser Information to your debug configuration, see Source Browser Information for a Configuration.
Page 4 of 28
SQUARES program reads formatted data from the file datafile.dat and displays the calculated results. With the source code shown below, it does not generate the expected results:
PROGRAM SQUARES INTEGER INARR(10), OUTARR(10), I, K ! Read the input array from the data file. OPEN(UNIT=8, FILE='datafile.dat', STATUS='OLD') READ(8,*,END=5) N, (INARR(I), I=1,N) 5 CLOSE (UNIT=8) ! Square all nonzero elements and store in OUTARR. K = 0 DO I = 1, N IF (INARR(I) .NE. 0) THEN ! Is the error in this DO loop? OUTARR(K) = INARR(I)**2 ENDIF END DO ! Print the squared output values. Then stop. PRINT 20, N 20 FORMAT (' Total number of elements read is',I4) PRINT 30, K 30 FORMAT (' Number of nonzero elements is',I4) DO, I=1,K PRINT 40, I, OUTARR(K) 40 FORMAT(' Element', I4, ' has value',I6) END DO END PROGRAM SQUARES
The formatted file datafile.dat currently contains one record that includes the following:
l l
An INTEGER count of the number of array elements, value 4 The values for each of the four INTEGER array elements
To view the values of this formatted data file in the Microsoft visual development environment, use the Open item in the File menu. When executed without array bounds checking (set by the /check:nobounds option), the output appears as follows:
When the program was executed with array bounds checking on, the output appears as follows:
Page 5 of 28
You can either build this program from the command line or within the visual development environment (see Preparing Your Program for Debugging). This example assumes a project workspace already exists. To debug this program: 1. From the Compaq Visual Fortran program folder, click Developer Studio to start the visual development environment. 2. In the File menu, click Open Workspace. 3. Click the FileView pane in the Workspace window. If the Workspace window is not displayed, click Workspace in the View menu. 4. Edit the file squares.f90: double-click its file name in the FileView pane. The screen appears as follows:
Page 6 of 28
To change the displayed toolbars, select Customize in the Tools menu and click the Toolbars tab. You can move toolbars by dragging the anchor (double vertical line on the left of the toolbar). 5. Click the first executable line to set the cursor position. In this case, click the beginning of the OPEN statement line:
OPEN(UNIT=8, FILE='datafile.dat', STATUS='OLD')
6. Click the Set/Remove Breakpoint (open hand symbol) button in the Build toolbar:
The red circle in the left margin of the text editor/debugger window shows where a breakpoint is set. 7. This example assumes you have previously built your application (see
Page 7 of 28
7. This example assumes you have previously built your application (see Preparing Your Program for Debugging). In the Build menu, click the Start Debug, Go item:
8. The debugger is now active. The current position is marked by a yellow arrow at the first executable line (the initial breakpoint):
Page 8 of 28
The Debug menu appears on the visual development environment title bar in place of the Build menu. If not displayed previously, the Debug toolbar appears. If needed, you can set another breakpoint, position the cursor at the line where you want to add or remove a breakpoint and do either of the following:
In the Build toolbar, click the Set/Remove Breakpoint button. In the Edit menu, click Breakpoints. A dialog box allows you to set or clear breakpoints, evaluate expressions, and perform other functions.
Step through the lines of source code. You can do this with the Debug menu item Step Over (as shown) or the Step Over button on the Debug toolbar:
9. Repeat the Step Over action and follow program execution into the DO loop. Repeat the Step Over action until you are at the end of the program. Position the cursor over the variable K to view its value (called Data Tips):
Page 9 of 28
The error seems related to the value of variable K! 10. In the text editor, add the line K = K + 1 as follows:
! Square all nonzero elements and store in OUTARR. K = 0 DO I = 1, N IF (INARR(I) .NE. 0) THEN K = K + 1 ! add this line OUTARR(K) = INARR(I)**2 ENDIF END DO
11. You have modified the source, so you need to rebuild the application: In the Debug menu, click Stop Debugging In the Build menu, click Build Squares.exe In the Build menu, click Execute Squares.exe or click the exclamation point (!) on the Build toolbar. The output screen appears as follows:
12. Although the program generates better results, you can examine the values of both the input array INARR (read from the file) and the output array OUTARR that the program calculates. In the text editor window, the previously set breakpoint remains set.
Page 10 of 28
In the Build menu, click the Start Debug, Go item. 13. To view the values of certain variables as the program executes, we need to display the Variables or the Watch window. In the View menu, click the Debug Windows, Variables window item:
14. In the Variables window, click the Locals tab to display the values of your local variables:
Page 11 of 28
You can view the values of the local variables by using the Locals tab, including the arrays (click the plus sign). The Variables window displays a Context menu (after the word Context:). The Context menu can help you debug exceptions. The Locals tab does not let you display module variables or other non-local variables. To display non-local variables, display the Watch window:
Page 12 of 28
15. Although this example does not use module variables or non-local variables, you can drag a variable name into the Watch window so the variable can be displayed. The Watch window allows you to display expressions. In the text editor window, select the variable name INARR (without its subscript syntax), drag it, and drop it into the Watch window:
16. Also drag the OUTARR array name to the Watch window. Click the Plus sign (+) to the left of the OUTARR variable's name to display the values of its array elements. 17. Execute lines of the program by using the Step Over button on the Debug toolbar. As the program executes, you can view the values of scalar variables with the data tips feature and view the values of arrays (or other variables) in the Watch window. When the program completes execution, the screen appears as follows:
Page 13 of 28
If a Disassembly window (shows disassembled code with source-code symbols) unintentionally appears, click the Step Out button on the debugger toolbar (or select the Step Out item in the Debug menu) as needed to dismiss the Disassembly window. If you have the Visual Fortran Professional or Enterprise Edition, you can use the Array Viewer to display and graph multidimensional array element values. For more information:
l l
On using breakpoints, see Using Breakpoints in the Debugger. On viewing different types of Fortran data, see Viewing Fortran Data Types in the Debugger. On displaying array values in the Array Viewer, see Using the Array Viewer in the Debugger. On locating errors in your program, see Locating Run-Time Errors in the Debugger. On additional debugger capabilities, see the Debugger section of the Visual C++ Users Guide.
Page 14 of 28
Viewing and Using Location Breakpoints Setting Location Condition Breakpoints Using Data Breakpoints (Current Scope) Setting Advanced Data Breakpoints
Viewing and Using Location Breakpoints To view the breakpoints currently set: 1. In the Edit menu, click Breakpoints. 2. Scroll up or down in the Breakpoints list to view the breakpoints. Enabled breakpoints have a check mark in the check box and disabled breakpoints have an empty check box. The Edit Breakpoints dialog box displays all types of breakpoints (not just location breakpoints). Alternatively, for smaller programs you can view location breakpoints set in the text editor window. Enabled location breakpoints are identified as a red circle in the left margin (see Debugging the Squares Example Program). Disabled location breakpoints are identified as a hollow circle in the left margin. You can also remove a breakpoint. To set a location breakpoint: 1. After you open the project workspace, open the source file within which you will set a location breakpoint in the text editor (double-click the file name in the FileView pane). 2. In a text editor window (or in the Call Stack or Disassembly window), click the line at which you want to enable a location breakpoint. 3. Do one of the following: Click the Insert/Remove Breakpoint button (open hand symbol) button in the Build toolbar or press the F9 key. When you set a breakpoint, it is enabled by default. Right click on that line, and select Insert/Remove Breakpoint from the pop-up menu. In the Edit menu, select Breakpoints. In the Location tab, click the arrow to the right of the Break At field. Click the displayed line number. Click OK. To disable or enable a location breakpoint: 1. In a text editor window (or in the Call Stack or Disassembly window), click the line containing the location breakpoint you want to disable. 2. Do one of the following: Right click on that line, and select Disable Breakpoint or Enable Breakpoint from the pop-up menu. In the Edit menu, select Breakpoints. Click on the check box for that
Page 15 of 28
breakpoint so it is unchecked (disabled) or checked (enabled). Click OK. To remove a location breakpoint: 1. In a text editor window (or in the Call Stack or Disassembly window), click the line containing the location breakpoint you want to remove. 2. Do one of the following: Click the Insert/Remove Breakpoint button (open hand symbol) button in the Build toolbar or press the F9 key. If the breakpoint was originally disabled, click the Insert/Remove Breakpoint button or press F9 again to remove it. Right click on that line, and select Remove Breakpoint from the popup menu. In the Edit menu, select Breakpoints. In the Location tab, click (select) the breakpoint to be removed and click the Remove button. Click OK. To view the source code where a location breakpoint is set: 1. In the Edit menu, select Breakpoints. 2. In the Breakpoints list, click (select) a location breakpoint. 3. Click the Edit Code button (upper right corner). This action takes you to the source code for a breakpoint set at a line number or function (or subroutine) name. In the case of function (or subroutine) names, the debugger must be running for this to work. To remove all breakpoints: 1. In the Edit menu, select Breakpoints. Click on the Remove All button. Click OK. Setting Location Condition Breakpoints Like a location breakpoint, a location condition breakpoint occurs only at a certain location. Instead of the break always occurring when the program executes at that location, a location condition breakpoint occurs when the program executes at that location and the specified condition is met (such as when the value of an array element is greater than 1). The following procedure applies to source code within the current scope. Unlike location breakpoints, location condition breakpoints must be set only using the Edit Breakpoints dialog box: To set a location condition breakpoint:
Page 16 of 28
1. In a text editor window (or in the Call Stack or Disassembly window), click the line at which you want to enable a location condition breakpoint. 2. Do the following: In the Edit menu, select Breakpoints. In the Location tab, click the arrow to the right of the Break At field. Click the displayed line number. Click the Condition button to display the Condition Breakpoint dialog box. Type the desired condition, such as:
A(1) .gt. 0
To disable, enable, and remove a location condition breakpoint, follow the general procedures for a location breakpoint. Under certain conditions, the debugger may disable the location condition breakpoint. In this case, you should either try to enable it or remove and set it again. When the program is run and if the specified condition occurs at the chosen location, a message box similar to the following appears:
Using Data Breakpoints (Current Scope) A data breakpoint displays a message box when the value of a variable changes or when a specified condition is met. Unlike a location condition breakpoint, data breakpoints are not associated with a specific source location. The following procedure applies to debugging the current routine (current scope). To set a data breakpoint: 1. In the Edit menu, select Breakpoints. 2. Click the Data tab. 3. Either type the variable name alone if you want a data breakpoint to occur whenever its value changes or type the desired condition, such as:
A(1) .eq. 0
Page 17 of 28
To disable, enable, or remove a data breakpoint, do one of the following: 1. In the Edit menu, select Breakpoints. 2. To disable or enable the data breakpoint, use the check box to the left of the data breakpoint (check indicates enabled). 3. To remove a data breakpoint, click (select) the data breakpoint and click the Remove button. Under certain conditions, the debugger may disable the data breakpoint. In this case, you should either try to enable it or remove and set it again. Setting Advanced Data Breakpoints You can request that your program stop execution when a specific variable is set to a specific logical expression (data breakpoint or watchpoint). To properly instruct the debugger to watch for a change of a certain data value (variable), you should specify the routine name (scope) within which the data value condition will be watched. When the data value condition for the variable occurs within the specified scope, a message box window is displayed informing you that the condition has been met. To set a data breakpoint for a specific local variable: 1. 2. 3. 4. In the Edit menu, select Breakpoints. Click the Data tab Enter the logical expression to be evaluated, such as K .GT. 2 Unless the data value being watched is in the main program code, specify the scope of the data breakpoint as follows: Click the right arrow (to the right of the expression) to display the Advanced pop-up menu:
Page 18 of 28
Click Advanced and specify the routine (function) name within which the data expression will be watched. Click OK 5. Start the debugger (Build menu). Consider displaying the source file associated with the data expression being watched. 6. If you have previously set an initial breakpoint for the program, in the Debug menu, click Go. Execution stops at the initial breakpoint. 7. In the Debug menu, click Go. Execution continues until the specified data value expression is met. A message box similar to the following appears when the specified data expression is met:
8. You can now view the source line just executed, examine current variable values, execute the next instruction, and so on to help you better understand that part of your program. If the error is related to an I/O statement, also see Viewing the Call Stack and Context Menu. Some limitations with the current implementation:
l l
Setting data breakpoints on array elements is not fully supported. Under certain conditions, before you run the program again, you may need remove the original data breakpoint and set the data breakpoint again.
For scalar (nonarray) data, use the data tips (leave pointer on a variable name) feature or use the Local Variables window. For single-dimension array data, derived-type data, record structure data, and COMPLEX data, use the Local Variables window or the Watch window. For multidimension array data, use the Local Variables window, the Watch window, or (Professional and Enterprise Editions) the Array Viewer. For common blocks in a DLL, move (drag) the name of the common block
Page 19 of 28
to the Watch window. You will be able to view the common block fields like any other structure. For information on using Data Tips, the Local Variables window, or a Watch window, see Debugging the Squares Example Program. For information on using the Array Viewer in the debugger, see Using the Array Viewer in the Debugger. The following sections apply to using the Watch window:
l l l
To display the Watch window: 1. In the View menu, point at (or click) Debug Windows 2. In the submenu, click Watch Specifying Array Sections You can specify array sections in a watch window. For example, consider an array declared as:
integer foo(10)
You can specify the following statement in a watch window to see the 2nd, 5th, and 8th elements:
foo(2:10:3)
When working with character arrays, this syntax may be combined with a substring specification. Consider the following array declaration:
character*8 chr_arr(10)
You can specify the following statement in a watch window to display the substring made up of character 3 through 8 of array elements 2 through 5:
chr_arr(2:5)(3:8)
This support is available for arrays of any type, including array pointers, assumed-shape, allocatable, and assumed-size arrays. Any valid integer expression can be used when specifying lower bound, upper bound, or stride. If the lower bound is omitted, the array lower bound is used. If the upper bound is omitted, the array upper bound is used. For example,
Page 20 of 28
To display:
l l l
Elements 1 through 8, specify foo(:8) Elements 5 through 10, specify foo(5:) All 10 elements, specify foo(:)
Specifying Module Variables To view a module variable in the Watch window, specify the module name, followed by "::", followed by the variable name. For example, to watch variable "bar" of module "foo", specify the following expression:
foo::bar
Specifying Format Specifiers You can use format specifiers in Watch windows to display variables in different data formats. For example, given a REAL variable 'foo' in a program, it is now possible to see 'foo' in different floating point notation (by typing "foo,f" "foo,g" or "foo,e" in a Watch window) or as an integer (""foo,i" or "foo,d"), a hexadecimal value ("foo,x"), an an octal value ("foo,o"), and so on. You can change the display format of variables in the Watch window using the formatting symbols in the following table: Symbol d,i o x,X f e g Format signed decimal integer unsigned octal integer Hexadecimal integer signed floating-point signed scientific notation signed floating-point or signed Value 0xF000F065 0xF065 61541 (decimal) 3./2. 3./2. 3./2. Displays -268373915 0170145 #0000F065 1.5000000 0.1500000E+01 1.500000
Debugging Fortran Programs scientific notation, whichever is shorter c s Single character String 0x0065 0x0012fde8 'e'
Page 21 of 28
"Hello world"
To use a formatting symbol, type the variable name, followed by a comma and the appropriate symbol. For example, if var has a value of 0x0065, and you want to see the value in character form, type var,c in the Name column on the tab of the Watch window. When you press ENTER, the character-format value appears:
var,c = 'e'
You can use the formatting symbols shown in the following table to format the contents of memory locations: Symbol ma m Format 64 ASCII characters 16 bytes in hexadecimal, followed by 16 ASCII characters 16 bytes in hexadecimal, followed by 16 ASCII characters 8 words 4 doublewords Displays
mb
0x0012ffac B3 34 CB 00 84 30 94 80 FF 22 8A 30 57 26 00 00 .4...0...".0W&..
mw md
0x0012ffac 34B3 00CB 3084 8094 22FF 308A 2657 0000 0x0012ffac 00CB34B3 80943084 308A22FF 00002657
With the memory location formatting symbols, you can type any value or expression that evaluates to a location. A formatting character can follow an expression also:
rep+1,x alps[0],mb
Page 22 of 28
Note: You can apply formatting symbols to structures, arrays, pointers, and objects as unexpanded variables only. If you expand the variable, the specified formatting affects all members. You cannot apply formatting symbols to individual members.
To use the Array Viewer in the debugger: 1. Start the debugger (see Debugging the Squares Example Program) 2. Step through the parts of the program that generate the values of the array you want to view, perhaps after stopping program execution at a certain breakpoint. 3. Select (click) the name of the array you want to view in the Array Viewer. For example:
Page 23 of 28
4. In the Debug menu, click Fortran Array Viewer or click the Array Viewer button in the Fortran toolbar:
To display the Fortran toolbar: 1. In the Tools menu, select Customize. 2. Click the Toolbars tab. 3. Set the Fortran toolbar check box.
Page 24 of 28
In the upper part (data window), Array Viewer displays the values of the array elements for VAL. In the lower part (graph window), the view shown for array VAL is a twodimensional view. You can also display other types of views, such as a height plot (in the View menu, click Height Plot). Initially, the Fortran Arrays window shows the current array being displayed and its status. 5. After your program changes the data in the Array being viewed, you can: Refresh the current Array Viewer data and graph by double-clicking the array name in the Fortran Array window. Create another instance of Array Viewer with an updated view of the data by clicking the Fortran Arrays button in the Fortran toolbar. The second instance may contain different value limits on the graph. 6. The Fortran Arrays window allows you to: To Do This: In the Fortran Arrays Window, Do This:
Page 25 of 28
Display a different array in the existing Array Viewer Display a different array in a new instance of the Array Viewer Update the displayed array's values Discontinue using the Array Viewer (removes the name from the Fortran Arrays window) Create a detached instance of Array Viewer (not associated with the debugger)
Click the Name column and type an array name Click in the last row of the Name column, type the array name, and press Enter (Return). A new instance of Array Viewer appears for the specified array. Double-click the line for the array Either stop the debugger (Debug menu, Stop Debugging item) or repeatedly select the array name and click the Delete key. Click on an array in the Name column and press the Escape (Esc) key. This removes the array name from the Fortran Arrays window and creates a detached instance of the Array Viewer that remains after the debugging session ends.
When using the Array Viewer, in the Help menu, click Help Topics. Read the Array Visualizer HTML Help documentation.
Effect of Certain Compiler Options Debugging an Exception Viewing the Call Stack and Using the Context Menu
Page 26 of 28
Be aware that your program must use compiler options that allow the debugger to catch the appropriate error conditions:
l
The /check:keyword options generate extra code to catch certain conditions at run-time. For example, if you do not specify Array and Substring Bounds checking (/check:bounds), the debugger will not catch and stop at array or character string bounds errors. If you specify the /fpe:3 compiler option, certain floating-point exceptions will not be caught, since this setting allows IEEE exceptional values and program continuation. In contrast, specifying /fpe:0 stops execution when an exceptional value (such as a NaN) is generated or when attempting to use a denormalized number, which usually allows you to localize the cause of the error.
In most cases, your program will automatically stop at the point where the exception occurs, allowing you to view the source code and values of variables. If the error is related to an I/O statement, see Viewing the Call Stack and Using the Context Menu. Debugging an Exception You can request that the program always stop when a certain type of exception occurs. Certain exceptions are caught by default by the Visual Fortran run-time library, so your program stops in the run-time library code. In most cases, you want the program to instead stop in your program's source code. To change how an exception is handled in the debugger: Start the debugger and stop execution at the initial breakpoint. In the Debug menu, click Exceptions View the displayed exceptions. Select each type of exception to be changed and change it from "Stop if not handled" to "Stop Always" 5. Continue program execution by clicking Go in the Debug menu. 6. When the exception occurs, you can now view the source line being executed, examine current variable values, execute the next instruction, and so on to help you better understand that part of your program. If the error is related to an I/O statement, also see Viewing the Call Stack and Using the Context Menu. 7. After you locate the error and correct the program, consider whether you want to reset the appropriate type of exception to "Stop if not handled" before you debug the program again. For machine exceptions, you can use the just-in-time debugging feature to debug your programs as they run outside of the visual development environment, if both of the following items have been set: 1. 2. 3. 4.
Page 27 of 28
In the Tools menu Options item, the Debug tab has the checkbox for JustIn Time debugging set. The FOR_IGNORE_EXCEPTIONS environment variable is set to TRUE.
Viewing the Call Stack and Using the Context Menu If you want to view where in the hierarchy of routines your program in currently executing, such as after your program stops at the point where an exception occurs, you can view the Call Stack window and Context menu in the debugger: 1. Start the debugger and stop at a breakpoint 2. In the View menu, click Debug Windows 3. In the submenu, click Call Stack A severe unhandled I/O programming error (such as an End-of-File condition) can occur while the program is executing in the debugger. When this occurs, the Fortran run-time system will raise a debug event automatically to stop program execution, allowing display of the Call Stack Display. When the severe unhandled I/O error occurs in the debugger:
l
Click OK to dismiss the information box. Scanning down the Call Stack display, there will be a few frames from NTDLL and the Fortran run-time system displayed, and then the actual Fortran routine with the I/O statement that caused the error. In the Context menu, select the Fortran routine to display the Fortran code. The green arrow points to the I/O statement that caused the error. You can view the Context menu (after Context:) to help locate the source code line that executed the I/O statement. The Context menu appears in the top of the Variables window (see Debugging the Squares Example Program). Use the Context menu to select the viewing context for the routine (use the arrow at the right to display selections). This action all occurs after the error message and traceback information has been displayed. The error message and traceback information is available in the program output window. To view the program output window, either iconize (minimize) the visual development environment or click the icon for the output window in the task bar. You should not need the stack dump because you have the Call Stack window in the visual development environment, but the error
Page 28 of 28
the Call Stack window in the visual development environment, but the error message with the file name might be useful to see. For more information:
l
On locating exceptions and the compiler options needed, see Locating RunTime Errors On using traceback information, see Using Traceback Information
Page 1 of 58
Software Environment and Efficient Compilation Important software environment suggestions that apply to nearly all applications, including using the most recent version of the compiler, related performance tools, and efficient ways to compile using the DF command or the visual development environment Analyze Program Performance Analyzing program performance, including using profiling tools Data Alignment Considerations Guidelines related to avoiding unaligned data Use Arrays Efficiently Guidelines for efficient array use Improve Overall I/O Performance Guidelines related to improving overall I/O performance Additional Source Code Guidelines for Run-Time Efficiency Additional performance guidelines related to source code Optimization Levels: the /optimize:num Option Understanding the compiler /optimize:num optimization level options and the types of optimizations performed Other Options Related to Optimization Understanding other compiler optimization options (besides the /optimize:num options) Compiler Directives Related to Performance Discuss the source directives (cDEC$ prefix) related to performance that can be used in place of compiler options or to provide more control of certain optimizations
Obtain and install the latest version of Visual Fortran, along with performance products that can improve application performance. Use the DF command and its options in a manner that lets the Visual Fortran compiler perform as many optimizations as possible to improve run-time performance. Use certain performance capabilities provided by the operating system.
Page 2 of 58
The latest version of Visual Fortran New releases of the Visual Fortran compiler and its associated run-time libraries may provide new features that improve run-time performance. For information on more recent Visual Fortran releases and other information about Visual Fortran, access the Compaq Fortran web page. If you have the appropriate technical support contract, you can also contact the Compaq technical support center for information on new releases (see "Visual Fortran Technical Support" in Compaq Visual Fortran Installing and Getting Started).
Performance profiling tools The visual development environment profiling tools allow function and line profiling. For more information on profiling, see Analyze Program Performance.
System-wide performance products Other products are not specific to a particular programming language or application, but can improve system-wide performance, such as minimizing disk device I/O and handling capacity planning. When running large programs, such as those accessing large arrays, adequate process limits and virtual memory (paging file) space as well as proper system tuning are especially important.
Page 3 of 58
During the later stages of program development (such as for the release configuration), you should:
l l
Avoid using incremental linking. Specify multiple source files together and use an optimization level of at least /optimize:4 on the DF command line to allow more interprocedural optimizations to occur. For instance, the following command compiles all three source files together using the default level of optimization (/optimize:4):
DF /exe:main.exe main.f90 sub2.f90 sub3.f90
When using the visual development environment, consider reducing the number of top-level source files by copying your application to a separate project. The separate project should use only one (or a few) top-level Fortran source files that contain INCLUDE statements for the other source files. Or, you can build your application from the command line, which allows many source files to be compiled together with a single command. For information on creating (exporting) makefile for command-line use, see Files in a Project; for information about using NMAKE, see Building Projects with NMAKE.
Compiling multiple source files together lets the compiler examine more code for possible optimizations, which results in:
l l l
Inlining more procedures More complete data flow analysis Reducing the number of external references to be resolved during linking
When compiling all source files together is not feasible (such as for very large programs), consider compiling related source files together using multiple DF commands rather than compiling source files individually. If you use the /compile_only option to prevent linking, also use the /object:file option so that multiple sources files are compiled into a single object file, allowing more optimizations to occur. Visual Fortran performs certain optimizations unless you specify the appropriate DF command-line options or corresponding visual development environment options in the Optimization category of the Fortran tab (see Categories of Compiler Options). Additional optimizations can be enabled or disabled using DF command options or in the visual development environment Project Settings dialog box Fortran tab. The following table shows DF options that can directly improve run-time performance on both ia32 and ia64 systems. Most of these options do not affect
Page 4 of 58
the accuracy of the results, while others improve run-time performance but can change some numeric results. Options Related to Run-Time Performance Option Names /align: keyword Description
Controls whether padding bytes are added between data derived-type data, and Compaq Fortran record structures naturally aligned.
/architecture: keyword
Requests code generation for a specific chip generation. O chip generations use new instructions that provide improv applications, but those instructions are not supported by o
/fast
Sets the following performance-related options: records, sequence), /architecture:host /assume:noaccura (which changes the default of /check:[no]power), and
/assume:noaccuracy_sensitive
Allows the compiler to reorder code based on algebraic id performance, enabling certain optimizations. The numeric different from the default (accuracy_sensitive) because o are rounded. This slight difference in numeric results is ac
/assume:buffered_io
Allows records that are otherwise written (flushed) to disk (default) to be accumulated in the buffer and written as a usually makes disk I/O more efficient by writing larger blo often.
/inline:all
Inlines every call that can possibly be inlined while genera recursive routines are not inlined to prevent infinite loops
/inline:speed
Inlines procedures that will improve run-time performanc increase in program size.
/inline:size
Inlines procedures that will improve run-time performanc in program size. This type of inlining occurs with optimiza or /optimize:5.
/math_library:fast
On ia32 systems, requests that arguments to the math lib to improve performance.
Page 5 of 58
Controls the optimization level and thus the types of optim default optimization level is /optimize:4 (except in the vis for a debug configuration), unless you specify /debug, wh to /optimize:0 (no optimizations). Use /optimize:5 to acti optimizations and (on ia64 systems) the software pipelini
Activates the software pipelining optimization (a subset o systems. Activates a group of loop transformation optimizations (a
Specifies the target processor generation (chip) architectu be run, allowing the optimizer to make decisions about in needed to create the most efficient code. Keywords allow processor generation type, multiple processor generation generation type currently in use during compilation. Rega keyword, the generated code compiled on ia32 systems w implementations of the ia32 architecture.
/unroll:num
Specifies the number of times a loop is unrolled ( level /optimize:3 or higher. If you omit /unroll: num many times loops are unrolled. Primarily on ia32 systems limit may improve run-time performance for certain applic
The following table lists options that can slow program performance on ia32 and ia64 systems. Some applications that require floating-point exception handling might need to use a different /fpe:n option. Other applications might need to use the /assume:dummy_aliases or /vms options for compatibility reasons. Other options listed in the table are primarily for troubleshooting or debugging purposes. Options that Slow Run-Time Performance Option Names /assume:dummy_aliases Description Forces the compiler to assume that dummy (formal) arguments to procedures share memory locations with other dummy arguments or with variables shared through use association, host association, or common block use. These program semantics slow performance, so For More Information See Dummy Aliasing Assumption
Performance: Making Programs Run Faster you should specify /assume: dummy_aliases only for the called subprograms that depend on such aliases. The use of dummy aliases violates the FORTRAN 77 and Fortran 90 standards but occurs in some older programs. /compile_only If you use /compile_only when compiling multiple source files, also specify /object:file to compile many source files together into one object file. Separate compilations prevent certain interprocedural optimizations, the same as using multiple DF commands or using /compile_only without the /object:file option. Generates extra code for array bounds checking at run time. Generates extra code to check integer calculations for arithmetic overflow at run time. Once the program is debugged, you may want to omit this option to reduce executable program size and slightly improve run-time performance. On ia32 systems, /fpe:3 provides the best performance. Using /fpe:0 slows program execution. On ia64 systems, using /fpe:0 provides the best performance. Using /fpe:3 slows program execution. /debug:full, /debug, or equivalent Generates extra symbol table information in the object file. Specifying /debug also reduces the default level of optimization to /optimize:0.
Page 6 of 58
/check:bounds /check:overflow
/fpe:n values
See /fpe
See /debug
Performance: Making Programs Run Faster /inline: none /inline: manual Prevents the inlining of all procedures (except statement functions). Reduces the optimization level (and types of optimizations). Use during the early stages of program development or when you will use the debugger. Controls certain VMS-related runtime defaults, including alignment. If you specify the /vms option, you may need to also specify the /align:records option to obtain optimal run-time performance.
Page 7 of 58 See Controlling the Inlining of Procedures See /[no] optimize and Optimization Levels: the /optimize Option See /[no]vms
/vms
Page 8 of 58
Run program timings when other users are not active. Your timing results can be affected by one or more CPU-intensive processes also running while doing your timings. Try to run the program under the same conditions each time to provide the most accurate results, especially when comparing execution times of a previous version of the same program. Use the same system (processor model, amount of memory, version of the operating system, and so on) if possible. If you do need to change systems, you should measure the time using the same version of the program on both systems, so you know each system's effect on your timings. For programs that run for less than a few seconds, run several timings to ensure that the results are not misleading. Certain overhead functions like loading DLLs might influence short timings considerably. If your program displays a lot of text, consider redirecting the output from the program. Redirecting output from the program will change the times reported because of reduced screen I/O.
Methods of Timing Your Application To perform application timings, use a version of the TIME command in a .BAT file (or the function timing profiling option). You might consider modifying the program to call routines within the program to measure execution time (possibly using conditionally compiled lines). For example:
l
Compaq Fortran intrinsic procedures, such as CPU_TIME, SYSTEM_CLOCK, DATE_AND_TIME, and TIME. Library routines, such as ETIME or TIME.
Visual Fortran programs created in a Windows 98, Windows Me, or Windows 95 development environment can be run and analyzed on Windows NT 4 or Windows 2000 systems. Whenever possible, perform detailed performance analysis on a system that closely resembles the system(s) that will used for actual application use. Sample Command Procedure that Uses TIME and Performance Monitor The following example shows a .BAT command procedure that uses the TIME command and the Performance Monitor (perfmon) tool available on Windows NT 4 and Windows 2000 systems. The kill command that stops the perfmon tool is included on the Windows NT Resource kit; if the kill tool is not available on your system, manually end the perfmon task by using the task manager. This .BAT procedure assumes that the program to be timed is myprog.exe. Before using this batch file, start the performance monitor to setup logging of
Performance: Making Programs Run Faster the statistics that you are interested in: 1. 2. 3. 4.
Page 9 of 58
the DOS prompt type: Perfmon the View menu, select Log the Edit menu, select Add to Log and select some statistics the Options menu, select Log. In the dialog box: Name the log file. The following .BAT procedure assumes that you have named the logfile myprog.log. Consider adjusting the Log Interval. As the last step, be sure to select "Start Log". 5. In the File menu, select Save Workspace to save the setup information. The following .BAT procedure assumes you have saved the workspace as my_perfmon_setup.pmw.
At In In In
rem Delete previous logs, then start up the Performance Monitor. rem We use start so that control returns instantly to this batch file. del myprog.log start perfmon my_perfmon_setup.pmw rem print the time we started time <nul | findstr current rem start the program we are interested in, this time using rem cmd /c so that the batch file waits for the program to finish. echo on cmd /c myprog.exe echo off rem print the time we stopped time <nul | findstr current rem all done logging statistics kill perfmon rem if kill is not available, end the perfmon task manually
After the run, analyze your data by using Performance Monitor: 1. 2. 3. 4. If it is not currently running, start Performance Monitor. In the View menu, select Chart. In the Options menu, select Data From and specify the name of the logfile. In the Edit menu, select Add To Chart to display the counters.
About the optimizations that improve application performance without source code modification, see Compile With Appropriate Options and
Page 10 of 58
Multiple Source Files. About profiling your application, see Profiling and Performance Tools.
l l l
The time spent in functions and the number of times a function was called (function timing). Only the number of times a function was called (function counting). A list of functions executed or not executed (function coverage). Information about the stack when each function is called (function attribution).
Function profiling does not require debug information (it obtains addresses from a .MAP file). Since function profiling (except function attribution) uses the stack, routines that modify the stack cannot be profiled. Exclude object files for routines that modify the stack. To perform function profiling: 1. In the Project menu, select Settings. 2. Click the Link tab. 3. In the General category, click the Enable profiling checkbox (this turns off
Performance: Making Programs Run Faster incremental linking). In the General category, click the Generate mapfile checkbox. Click OK to accept the current project settings. Build your application. After building your application, profile your project.
Page 11 of 58
4. 5. 6. 7.
Line Profiling Line profiling collects more information than function profiling. It shows how many times a line is executed and whether certain lines are not executed. Line profiling requires debug information. To perform line profiling: 1. In the Project menu, select Settings. 2. Click the Link tab. 3. In the General category, click the Enable profiling checkbox (this turns off incremental linking). 4. In the General category, click the Generate debug information checkbox. 5. Click on the Fortran tab. 6. In the category drop-down list, select Debug. 7. In the Debugging level drop-down list, select Full. 8. In the Debugging level drop-down list, click the Use Program Database for Debug Information checkbox. 9. Click OK to accept the current project settings. 10. Build your application 11. After building your application, profile your project. Performance Tools Tools that you can use to analyze performance include:
l l l
Process Viewer (Pview) lets you view process and thread characteristics. Spy++ provides a graphical view of system use. On Windows NT or Windows 2000 systems, the Windows Performance Monitor can help identify performance bottlenecks. Other performance tools are available in the Microsoft Win32 SDK (see the online Platform SDK Tools Guide, Tuning section in HTML Help Viewer).
You can also purchase separate products to perform performance analysis and profiling. Efficient Source Code Once you have determined those sections of code where most of the program execution time is spent, examine these sections for coding efficiency. Suggested guidelines for improving source code efficiency are provided in the following
Page 12 of 58
Data Alignment Considerations Use Arrays Efficiently Improve Overall I/O Performance Additional Source Code Guidelines for Run-Time Efficiency
For information about timing your application and for an example command procedure that uses the Windows NT Performance Monitor, see Timing Your Application.
Causes of Unaligned Data and Ensuring Natural Alignment Checking for Inefficient Unaligned Data Ordering Data Declarations to Avoid Unaligned Data Options Controlling Alignment
Page 13 of 58
Common blocks (COMMON statement) The order of variables in the COMMON statement determines their storage order. Unless you are sure that the data items in the common block will be naturally aligned, specify either the /align:commons or /align:dcommons option, depending on the largest data size used. For examples and more information, see Arranging Data in Common Blocks.
Derived-type (user-defined) data Derived-type data members are declared after a TYPE statement. If your data includes derived-type data structures, unless you are sure that the data items in derived-type data structures will be naturally aligned, use the /align:records option (default). If you omit the SEQUENCE statement, the /align:records option ensures all data items are naturally aligned. If you specify the SEQUENCE statement, the /align:records option is prevented from adding necessary padding to avoid unaligned data (data items are packed) unless you also specify the /align:sequence option. When you use SEQUENCE, you should specify a data declaration order such that all data items are naturally aligned or specify /align:sequence. For an example and more information, see Arranging Data Items in Derived-Type Data.
Compaq Fortran record structures (RECORD and STRUCTURE statements) Compaq Fortran record structures usually contain multiple data items. The order of variables in the STRUCTURE statement determines their storage order. The RECORD statement names the record structure.
Page 14 of 58
If your data includes Compaq Fortran record structures, you should use the /align:records option (default), unless you are sure that the data items in derived-type data and Compaq Fortran record structures will be naturally aligned. For examples and more information, see Arranging Data Items in Compaq Fortran Record Structures.
l
Equivalenced data EQUIVALENCE statements can force unaligned data or cause data to span natural boundaries.
To avoid unaligned data in a common block, derived-type data, or record structures, use one or both of the following:
l
For new programs or for programs where the source code declarations can be modified easily, plan the order of data declarations with care. For example, you should order variables in a COMMON statement such that numeric data is arranged from largest to smallest, followed by any character data (see the data declaration rules in Ordering Data Declarations to Avoid Unaligned Data). Consider using explicit-length declarations, such as INTEGER(KIND=8) instead of INTEGER, which uses default kind. This prevents command-line options like /real_size and /integer_size from changing the length of INTEGER, LOGICAL, REAL, and COMPLEX declarations, which may change alignment. For existing programs where source code changes are not easily done or for array elements containing derived-type or record structures, you can use command line options to request that the compiler align numeric data by adding padding spaces where needed.
Other possible causes of unaligned data include unaligned actual arguments and arrays that contain a derived-type structure or Compaq Fortran record structure. When actual arguments from outside the program unit are not naturally aligned, unaligned data access will occur. Compaq Fortran assumes all passed arguments are naturally aligned and has no information at compile time about data that will be introduced by actual arguments during program execution. For arrays where each array element contains a derived-type structure or Compaq Fortran record structure, the size of the array elements may cause some elements (but not the first) to start on an unaligned boundary. Even if the data items are naturally aligned within a derived-type structure without the SEQUENCE statement or a record structures, the size of an array
Page 15 of 58
element might require use of /align options to supply needed padding to avoid some array elements being unaligned. If you specify /align:norecords, no padding bytes are added between array elements. If array elements each contain a derived-type structure with the SEQUENCE statement, array elements are packed without padding bytes regardless of the DF command options specified. In this case, some elements will be unaligned. When /align:records option is in effect, the number of padding bytes added by the compiler for each array element is dependent on the size of the largest data item within the structure. The compiler determines the size of the array elements as an exact multiple of the largest data item in the derived-type structure without the SEQUENCE statement or a record structure. The compiler then adds the appropriate number of padding bytes. For instance, if a structure contains an 8-byte floating-point number followed by a 3-byte character variable, each element contains five bytes of padding (16 is an exact multiple of 8). However, if the structure contains one 4-byte floatingpoint number, one 4-byte integer, followed by a 3-byte character variable, each element would contain one byte of padding (12 is an exact multiple of 4). For more information: On the /align options, see Options Controlling Alignment.
Double-check data declarations within common block, derived-type data, or record structures to ensure all data items are naturally aligned (see the data declaration rules in Ordering Data Declarations to Avoid Unaligned Data). Using modules to contain data declarations can ensure consistent alignment and use of such data. Avoid the EQUIVALENCE statement or use it in a manner that cannot cause unaligned data or data spanning natural boundaries. Ensure that passed arguments from outside the program unit are naturally aligned. Check that the size of array elements containing at least one derived-type data or record structure cause array elements to start on aligned boundaries (see Causes of Unaligned Data and Ensuring Natural Alignment).
Page 16 of 58
During compilation, warning messages are issued for any data items that are known to be unaligned (unless you specify the /warn:noalignments option).
Always define the largest size numeric data items first. If your data includes a mixture of character and numeric data, place the numeric data first. Add small data items of the correct size (or padding) before otherwise unaligned data to ensure natural alignment for the data that follows.
When declaring data, consider using explicit length declarations, such as specifying a KIND parameter. For example, specify INTEGER(KIND=4) (or INTEGER(4)) rather than INTEGER. If you do use a default length (such as INTEGER, LOGICAL, COMPLEX, and REAL), be aware that the compiler options /integer_size:num and /real_size:num can change the size of an individual field's data declaration size and thus can alter the data alignment of a carefully-planned order of data declarations. Using the suggested data declaration guidelines minimizes the need to use the /align options to add padding bytes to ensure naturally aligned data. In cases where the /align options are still needed, using the suggested data declaration guidelines can minimize the number of padding bytes added by the compiler:
l l l
Arranging Data Items in Common Blocks Arranging Data Items in Derived-Type Data Arranging Data Items in Compaq Fortran Record Structures
Page 17 of 58
As shown in the following figure, if you omit the alignment compiler options, the common block will contain unaligned data items beginning at the first array element of IARRY_I. Common Block with Unaligned Data
As shown in the following figure, if you compile the program units that use the common block with the /align:commons options, data items will be naturally aligned. Common Block with Naturally Aligned Data
Because the common block X contains data items whose size is 32 bits or smaller, specify /align:commons. If the common block contains data items whose size might be larger than 32 bits (such as REAL (KIND=8) data), use /align:dcommons. If you can easily modify the source files that use the common block data, define the numeric variables in the COMMON statement in descending order of size and place the character variable last to provide more portability and ensure natural alignment without padding or the DF command options /align:commons or /align:dcommons:
LOGICAL (KIND=2) FLAG INTEGER IARRY_I(3) CHARACTER(LEN=5) NAME_CH COMMON /X/ IARRY_I(3), FLAG, NAME_CH
As shown in the following figure, if you arrange the order of variables from largest to smallest size and place character data last, the data items will be naturally aligned. Common Block with Naturally Aligned Reordered Data
Page 18 of 58
When modifying or creating all source files that use common block data, consider placing the common block data declarations in a module so the declarations are consistent. If the common block is not needed for compatibility (such as file storage or Compaq Fortran 77 use), you can place the data declarations in a module without using a common block.
If you specify the SEQUENCE statement, the first data item is in the first storage location and the last data item is in the last storage location. The data items appear in the order in which they are declared. If you omit the /align:sequence option, data declarations must be carefully specified to naturally align data. If you omit the SEQUENCE statement, Compaq Fortran adds the padding bytes needed to naturally align data item components, unless you specify the /align:norecords option.
As shown in the following figure, the largest numeric data items are defined first
Page 19 of 58
and the character data type is defined last. There are no padding characters between data items and all items are naturally aligned. The trailing padding byte is needed because CATALOG_SPRING is an array; it is inserted by the compiler when the /align:records option is in effect. Derived-Type Naturally Aligned Data (in CATALOG_SPRING())
The following figure shows the memory diagram of record REC for naturally aligned records. Memory Diagram of REC for Naturally Aligned Records
Page 20 of 58
The /align:commons option Requests that data in common blocks be aligned on up to 4-byte boundaries, by adding padding bytes as needed. Unless you specify /fast, the default is /align:nocommons or arbitrary byte alignment of common block data. In this case, unaligned data can occur unless the order of data items specified in the COMMON statement places the largest numeric data item first, followed by the next largest numeric data (and so on), followed by any character data.
The /align:dcommons option Requests that data in common blocks be aligned on up to 8-byte boundaries, by adding padding bytes as needed. Unless you specify /fast, the default is /align:nocommons or arbitrary byte alignment of data items in a common data. Specify the /align:dcommons option for applications that use common blocks, unless your application has no unaligned data or, if the application might have unaligned data, all data items are four bytes or larger. For applications that use common blocks where all data items are four bytes or smaller, you can specify /align:commons instead of /align:dcommons.
The /align:norecords option Requests that multiple data items in derived-type data and record structures (a Compaq Fortran extension) be aligned arbitrarily on byte boundaries instead of being naturally aligned. The default is /align:records.
The /align:records option Requests that multiple data items in record structures and derived-type data without the SEQUENCE statement be naturally aligned, by adding padding bytes as needed. The default is /align:records.
The /align:sequence option Controls alignment of derived types with the SEQUENCE statement. Specifying the /align:sequence option means that derived types with the SEQUENCE statement obey whatever alignment rules are currently in use.
Page 21 of 58
Consequently, since /align:records is a default value, then /align:sequence alone on the command line will cause the fields in these derived types to be naturally aligned. Specifying /fast implies /align:sequence.
l
The /align:nosequence option Controls alignment of derived types with the SEQUENCE statement. The default /align:nosequence option means that derived types with the SEQUENCE statement are packed regardless of any other alignment rules.
The default behavior is that multiple data items in derived-type data and record structures will be naturally aligned; data items in common blocks will not be naturally aligned (/align:records) with /align:nocommons. In derived-type data, using the SEQUENCE statement without specifying /align:sequence prevents /align:records from adding needed padding bytes to naturally align data items.
The fastest array access occurs when contiguous access to the whole array or most of an array occurs. Perform one or a few array operations that access all of the array or major parts of an array rather than numerous operations on scattered array elements. Rather than use explicit loops for array access, use elemental array operations, such as the following line that increments all elements of array variable A:
A = A + 1.
When reading or writing an array, use the array name and not a DO loop
Page 22 of 58
or an implied DO-loop that specifies each element number. Fortran 90 array syntax allows you to reference a whole array by using its name in an expression. For example:
REAL :: A(100,100) A = 0.0 A = A + 1. . . . WRITE (8) A
Similarly, you can use derived-type array structure components, such as:
TYPE X INTEGER A(5) END TYPE X . . . TYPE (X) Z WRITE (8) Z%A
l
Make sure multidimensional arrays are referenced using proper array syntax and are traversed in the "natural" ascending order column major for Fortran. With column-major order, the leftmost subscript varies most rapidly with a stride of one. Whole array access uses column-major order. Avoid row-major order, as is done by C, where the rightmost subscript varies most rapidly. For example, consider the nested DO loops that access a two-dimension array with the J loop as the innermost loop:
INTEGER X(3,5), Y(3,5), I, J Y = 0 DO I=1,3 ! I outer loop varies slowest DO J=1,5 ! J inner loop varies fastest X (I,J) = Y(I,J) + 1 ! Inefficient row-major storage order END DO ! (rightmost subscript varies fastest) END DO . . . END PROGRAM
Because J varies the fastest and is the second array subscript in the expression X (I,J), the array is accessed in row-major order. To make the array accessed in natural column-major order, examine the array algorithm and data being modified. Using arrays X and Y, the array can be accessed in natural column-major order by changing the nesting order of the DO loops so the innermost loop
Performance: Making Programs Run Faster variable corresponds to the leftmost array dimension:
INTEGER Y = 0 X(3,5), Y(3,5), I, J ! ! ! !
Page 23 of 58
J outer loop varies slowest I inner loop varies fastest Efficient column-major storage order (leftmost subscript varies fastest)
Fortran whole array access (X= Y + 1) uses efficient column major order. However, if the application requires that J vary the fastest or if you cannot modify the loop order without changing the results, consider modifying the application program to use a rearranged order of array dimensions. Program modifications include rearranging the order of:
Dimensions in the declaration of the arrays X(5,3) and Y(5,3) The assignment of X(J,I) and Y(J,I) within the DO loops All other references to arrays X and Y
In this case, the original DO loop nesting is used where J is the innermost loop:
INTEGER X(5,3), Y(5,3), I, J Y = 0 DO I=1,3 ! I outer loop varies slowest DO J=1,5 ! J inner loop varies fastest X (J,I) = Y(J,I) + 1 ! Efficient column-major storage order END DO ! (leftmost subscript varies fastest) END DO . . . END PROGRAM
Code written to access multidimensional arrays in row-major order (like C) or random order can often make inefficient use of the CPU memory cache. For more information on using natural storage order during record I/O operations, see Write Array Data in the Natural Storage Order.
l
Use the available Fortran 95/90 array intrinsic procedures rather than creating your own. Whenever possible, use Fortran 95/90 array intrinsic procedures instead of creating your own routines to accomplish the same task. Fortran 95/90 array intrinsic procedures are designed for efficient use with the various Visual Fortran run-time components.
Page 24 of 58
Using the standard-conforming array intrinsics can also make your program more portable.
l
With multidimensional arrays where access to array elements will be noncontiguous, avoid left-most array dimensions that are a power of two (such as 256, 512). At higher levels of optimization (/optimize=3 or higher), the compiler pads certain power-of-two array sizes to minimize possible inefficient use of the cache. Because the cache sizes are a power of two, array dimensions that are also a power of two may make inefficient use of cache when array access is noncontiguous. One work-around is to increase the dimension to allow some unused elements, making the leftmost dimension larger than actually needed. For example, increasing the leftmost dimension of A from 512 to 520 would make better use of cache:
REAL A (512,100) DO I = 2,511 DO J = 2,99 A(I,J)=(A(I+1,J-1) + A(I-1, J+1)) * 0.5 END DO END DO
In this code, array A has a leftmost dimension of 512, a power of two. The innermost loop accesses the rightmost dimension (row major), causing inefficient access. Increasing the leftmost dimension of A to 520 (REAL A (520,100)) allows the loop to provide better performance, but at the expense of some unused elements. Because loop index variables I and J are used in the calculation, changing the nesting order of the DO loops changes the results.
l
To minimize data storage and memory cache misses with arrays, use 32bit data rather than 64-bit data, unless you require the greater range and precision of double precision floating-point numbers or the numeric range of 8-byte integers.
For more information: On arrays and their data declaration statements, see Arrays.
Page 25 of 58
Explicit-shape arrays were used with FORTRAN 77. These arrays have a fixed rank and extent that is known at compile time. Other dummy argument (receiving) arrays that are not deferred-shape (such as assumed-size arrays) can be grouped with explicit-shape array arguments in the following discussion. Deferred-shape arrays were introduced with Fortran 90. Types of deferredshape arrays include array pointers and allocatable arrays. Assumed-shape array arguments generally follow the rules about passing deferred-shape array arguments.
When passing arrays as arguments, either the starting (base) address of the array or the address of an array descriptor is passed:
l
When using explicit-shape (or assumed-size) arrays to receive an array, the starting address of the array is passed. When using deferred-shape or assumed-shape arrays to receive an array, the address of the array descriptor is passed (the compiler creates the array descriptor).
Passing an assumed-shape array or array pointer to an explicit-shape array can slow run-time performance, since the compiler needs to create an array temporary for the entire array. The array temporary is created because the passed array may not be contiguous and the receiving (explicit-shape) array requires a contiguous array. When an array temporary is created, the size of the passed array determines whether the impact on slowing run-time performance is slight or severe. The following table summarizes what happens with the various combinations of array types. The amount of run-time performance inefficiency depends on the size of the array. Output Argument Array Types Input Argument Array Types Explicit-Shape Arrays Explicit-Shape Arrays Deferred-Shape and Assumed-Shape Arrays Efficient. Only allowed for assumed-shape arrays (not deferred-shape arrays). Does not use an array temporary. Passes an array descriptor. Requires an interface block.
Very efficient. Does not use an array temporary. Does not pass an array descriptor. Interface block optional.
Performance: Making Programs Run Faster DeferredShape and AssumedShape Arrays When passing an allocatable array, very efficient. Does not use an array temporary. Does not pass an array descriptor. Interface block optional. When not passing an allocatable array, not efficient. Instead use allocatable arrays whenever possible. Uses an array temporary. Does not pass an array descriptor. Interface block optional.
Page 26 of 58 Efficient. Requires an assumed-shape or array pointer as dummy argument. Does not use an array temporary. Passes an array descriptor. Requires an interface block.
A dramatic reduction in CPU time without a corresponding improvement in I/O time results in an I/O bottleneck. By such coding practices as: Unnecessary formatting of data and other CPU-intensive processing Unnecessary transfers of intermediate results Inefficient transfers of small amounts of data Application requirements
Improved coding practices can minimize actual device I/O, as well as the actual CPU time. You can also consider solutions to system-wide problems like minimizing device I/O delays. The following sections discuss I/O performance considerations in more detail:
l l l
Use Unformatted Files Instead of Formatted Files Write Whole Arrays or Strings Write Array Data in the Natural Storage Order
Page 27 of 58
Use Memory for Intermediate Results Enable Implied-DO Loop Collapsing Use of Variable Format Expressions Efficient Use of Record Buffers and Disk I/O Specify RECL Use the Optimal Record Type
than S2:
S1 S2
100
Although formatted data files are more easily ported to other systems, Visual Fortran can convert unformatted data in several formats (see Converting Unformatted Numeric Data).
Performance: Making Programs Run Faster efficient block moves are inhibited.
Page 28 of 58
If the whole array is not being written, natural storage order is the best order possible. If you must use an unnatural storage order, in certain cases it might be more efficient to transfer the data to memory and reorder the data before performing the I/O operation.
The control variable must be an integer. The control variable cannot be a dummy argument or contained in an EQUIVALENCE or VOLATILE statement. Compaq Fortran must be able to determine that the control variable does not change unexpectedly at run time. The format must not contain a variable format expression.
See the VOLATILE attribute and statement. On loop optimizations, see Optimization Levels: the /optimize:num Option.
Page 29 of 58
time formatting, but they are more efficient because the compiler can eliminate run-time parsing of the I/O format. Only a small amount of processing and the actual data transfer are required during run time. On the other hand, run-time formatting can impair performance significantly. For example, in the following statements, S1 is more efficient than S2 because the formatting is done once at compile time, not at run time:
S1 400 WRITE (6,400) (A(I), I=1,N) FORMAT (1X, <N> F5.2) . . . WRITE (CHFMT,500) '(1X,',N,'F5.2)' FORMAT (A,I3,A) WRITE (6,FMT=CHFMT) (A(I), I=1,N)
S2 500
Page 30 of 58
Fortran run-time system empties its internal buffer for each WRITE (or similar record output statement). If you specify BUFFERED='YES', specify /assume:buffered_io, or at run-time set the FORT_BUFFERED environment variable to TRUE, for disk devices the internal buffer will be filled, possibly by many record output statements (WRITE), before it is written to disk. The OPEN statement BUFFERED specifier takes precedence over the /assume: [no]buffered_io option. If neither are set (default), the FORT_BUFFERED environment variable is tested at run-time. The OPEN statement BUFFERED specifier applies to a specific logical unit. In contrast, the /assume:[no]buffered_io option and the FORT_BUFFERED environment variable apply to all Fortran units. Using buffered writes usually makes disk I/O more efficient by writing larger blocks of data to the disk less often. However, a system failure when using buffered writes can cause records to be lost, since they might not yet have been written to disk (such records would have written to disk with the default unbuffered writes).
Specify RECL
The sum of the record length (RECL specifier in an OPEN statement) and its overhead is a multiple or divisor of the blocksize, which is device specific. For example, if the BLOCKSIZE is 8192 then RECL might be 24576 (a multiple of 3) or 1024 (a divisor of 8). The RECL value should fill blocks as close to capacity as possible (but not over capacity). Such values allow efficient moves, with each operation moving as much data as possible; the least amount of space in the block is wasted. Avoid using values larger than the block capacity, because they create very inefficient moves for the excess data only slightly filling a block (allocating extra memory for the buffer and writing partial blocks are inefficient). The RECL value unit for formatted files is always 1-byte units. For unformatted files, the RECL unit is 4-byte units, unless you specify the /assume:byterecl option to request 1-byte units. When porting unformatted data files from non-Compaq systems, see Converting Unformatted Numeric Data.
Page 31 of 58
For sequential files of a consistent record size, the fixed-length record type gives the best performance. For sequential unformatted files when records are not fixed in size, the variable-length record type gives the best performance, particularly for BACKSPACE operations. For sequential formatted files when records are not fixed in size, the Stream_LF record type gives the best performance.
On OPEN statement specifiers and defaults, see OPEN Statement and OPEN. On Visual Fortran data files, see Devices and Files.
Avoid Small Integer and Small Logical Data Items (ia64 only) Avoid Mixed Data Type Arithmetic Expressions Use Efficient Data Types Avoid Using Slow Arithmetic Operators Avoid EQUIVALENCE Statement Use Use Statement Functions and Internal Subprograms Code DO Loops for Efficiency
Avoid Small Integer and Small Logical Data Items (ia64 only)
To minimize data storage and memory cache misses with arrays, use 32-bit data rather than 64-bit data, unless you require the greater range and precision of double precision floating-point numbers or the numeric range of 8-byte integers. On ia64 systems, avoid using integer or logical data less than 32 bits (KIND=4).
Page 32 of 58
Accessing a 16-bit (KIND=2) or 8-bit (KIND=1) data type can result in a sequence of machine instructions to access the data, rather than a single, efficient machine instruction for a 32-bit data item.
For applications with numerous floating-point operations, consider using the /assume: accuracy_sensitive option (see Arithmetic Reordering Optimizations) if a small difference in the result is acceptable. You can use different sizes of the same general data type in an expression with minimal or no effect on run-time performance. For example, using REAL, DOUBLE PRECISION, and COMPLEX floating-point numbers in the same floatingpoint arithmetic expression has minimal or no effect on run-time performance.
On ia32 systems: Integer of four bytes, expressed as INTEGER(4) or INTEGER*4 Single-precision real, expressed explicitly as REAL, REAL(KIND=4), or REAL*4 Double-precision real, expressed explicitly as DOUBLE PRECISION, REAL(KIND=8), or REAL*8 Integer of eight bytes, expressed as INTEGER(8) or INTEGER*8 On ia64 systems:
Page 33 of 58
Integer of four or eight bytes (also see Avoid Small Integer and Small Logical Data Items (ia64 only)) Single-precision real, expressed explicitly as REAL, REAL(KIND=4), or REAL*4 Double-precision real, expressed explicitly as DOUBLE PRECISION, REAL(KIND=8), or REAL*8
However, keep in mind that in an arithmetic expression, you should avoid mixing integer and floating-point (REAL) data (see Avoid Mixed Data Type Arithmetic Expressions).
Addition (+), subtraction (-), and floating-point multiplication (*) Integer multiplication (*) Division (/) Exponentiation (**)
Force unaligned data or cause data to span natural boundaries. Prevent certain optimizations, including: Global data analysis under certain conditions (see Global Optimizations) Implied-DO loop collapsing when the control variable is contained in an EQUIVALENCE statement
Page 34 of 58
statement functions and internal subprograms maximizes the number of subprogram references that will be inlined, especially when multiple source files are compiled together at optimization level /optimize:4 or /optimize:5 (or an appropriate /inline keyword was specified). For more information, see Compile With Appropriate Options and Multiple Source Files.
On loop optimizations, see Loop Transformations, and Controlling Loop Unrolling, and Software Pipelining (ia64 only). On the DO statement, see DO in the A-Z Summary in the Language Reference
Page 35 of 58
Levels of Optimization with Different /optimize:num Options Optimization Type Software pipelining Loop transformation Automatic inlining Additional global optimizations Global optimizations Local (minimal) optimizations x x x x
/optimize:0
/optimize:1
/optimize:2
/optimize:3
/optimize:4
The default for the command line and in the visual development environment for a release configuration is /optimize:4. In the visual development environment for a debug configuration, the default is /optimize:0. However, when /debug is specified, the default is /optimize:0 (no optimizations). In the table, the following terms are used to describe the levels of optimization (described in detail in the following sections:
l
Local (minimal) optimizations (/optimize:1) or higher occur within the source program unit and include recognition of common subexpressions and the expansion of multiplication and division. Global optimizations (/optimize:2) or higher include such optimizations as data-flow analysis, code motion, strength reduction, split-lifetime analysis, and instruction scheduling. Additional global optimizations (/optimize:3) or higher improve speed at the cost of extra code size. These optimizations include loop unrolling, code replication to eliminate branches, and padding certain power-of-two array sizes for more efficient cache use. Automatic inlining (/optimize:4) or higher applies interprocedure analysis and inline expansion of small procedures, usually by using heuristics that limit extra code size.
Page 36 of 58
Loop transformation and software pipelining (/optimize:5), include a group of loop transformation optimizations and, on ia64 systems, also include the software pipelining optimization. The loop transformation optimizations apply to array references within loops and can apply to multiple nested loops. Loop transformation optimizations can improve the performance of the memory system. On ia64 systems, software pipelining applies instruction scheduling to certain innermost loops, allowing instructions within a loop to "wrap around" and execute in a different iteration of the loop. This can reduce the impact of long-latency operations, resulting in faster loop execution. Software pipelining also enables the prefetching of data to reduce the impact of cache misses.
Optimizations Performed at All Optimization Levels Local (Minimal) Optimizations Global Optimizations Additional Global Optimizations Automatic Inlining Loop Transformation and Software Pipelining
Space optimizations Space optimizations decrease the size of the object or executing program by eliminating unnecessary use of memory, thereby improving speed of execution and system throughput. Visual Fortran space optimizations are as follows:
Constant Pooling Only one copy of a given constant value is ever allocated memory space. If that constant value is used in several places in the program, all references point to that value.
Dead Code Elimination If operations will never execute or if data items will never be used, Visual Fortran eliminates them. Dead code includes unreachable code and code that becomes unused as a result of other optimizations, such as value propagation.
Page 37 of 58
Inlining arithmetic statement functions and intrinsic procedures Regardless of the optimization level, Visual Fortran inserts arithmetic statement functions directly into a program instead of calling them as functions. This permits other optimizations of the inlined code and eliminates several operations, such as calls and returns or stores and fetches of the actual arguments. For example:
SUM(A,B) = A+B . . . Y = 3.14 X = SUM(Y,3.0)
Many intrinsic procedures are automatically inlined. Inlining of other subprograms, such as contained subprograms, occurs at optimization level /optimize:4 or /optimize:5 (or when you specify appropriate /inline keywords at /optimize:1 or higher).
l
Implied-DO loop collapsing DO loop collapsing reduces a major overhead in I/O processing. Normally, each element in an I/O list generates a separate call to the Visual Fortran RTL. The processing overhead of these calls can be most significant in implied-DO loops. If Visual Fortran can determine that the format will not change during program execution, it replaces the series of calls in up to seven nested implied-DO loops with a single call to an optimized RTL routine (see Enable Implied-Do Loop Collapsing). The optimized RTL routine can transfer many elements in one operation. Visual Fortran collapses implied-DO loops in formatted and unformatted I/O operations, but it is more important with unformatted I/O, where the cost of transmitting the elements is a higher fraction of the total cost.
Array temporary elimination and FORALL statements Certain array store operations are optimized. For example, to minimize the creation of array temporaries, Visual Fortran can detect when no overlap occurs between the two sides of an array assignment. This type of optimization occurs for some assignment statements in FORALL constructs. Certain array operations are also candidates for loop unrolling optimizations (see Loop Unrolling).
Page 38 of 58
Common Subexpression Elimination Integer Multiplication and Division Expansion Compile-Time Operations Value Propagation Dead Store Elimination Register Usage Mixed Real/Complex Operations
Variables t1 and t2 represent equivalent expressions. Visual Fortran eliminates this redundancy by producing the following:
t = ((J-1)*25+(I-1))*4 A(t) = B(t)
Page 39 of 58
The optimizer uses machine code that, like the ISHFT intrinsic function, shifts bits to expand multiplication and division by literals.
Compile-Time Operations
Visual Fortran does as many operations as possible at compile time rather than having them done at run time. Constant Operations Visual Fortran can perform many operations on constants (including PARAMETER constants):
l l
Constants preceded by a unary minus sign are negated. Expressions involving +, -, *, or / operators are evaluated; for example:
PARAMETER (NN=27) I = 2*NN+J ! Becomes: I = 54 + J
Evaluation of some constant functions and operators is performed at compile time. This includes certain functions of constants, concatenation of string constants, and logical and relational operations involving constants.
l
Lower-ranked constants are converted to the data type of the higherranked operand:
REAL X, Y X = 10 * Y ! Becomes: X = 10.0 * Y
Array address calculations involving constant subscripts are simplified at compile time whenever possible:
INTEGER I(10,10) I(1,2) = I(4,5) ! Compiled as a direct load and store
Algebraic Reassociation Optimizations Visual Fortran delays operations to see whether they have no effect or can be transformed to have no effect. If they have no effect, these operations are removed. A typical example involves unary minus and .NOT. operations:
X = -Y * -Z ! Becomes: Y * Z
Value Propagation
Visual Fortran tracks the values assigned to variables and constants, including those from DATA statements, and traces them to every place they are used. Visual Fortran uses the value itself when it is more efficient to do so.
Page 40 of 58
When compiling subprograms, Visual Fortran analyzes the program to ensure that propagation is safe if the subroutine is called more than once. Value propagation frequently leads to more value propagation. Visual Fortran can eliminate run-time operations, comparisons and branches, and whole statements. In the following example, constants are propagated, eliminating multiple operations from run time: Original Code
PI = 3.14 . . . PIOVER2 = PI/2 . . . I = 100 . . . IF (I.GT.1) GOTO 10 10 10 A(I) = 3.0*Q
Optimized Code
. . . PIOVER2 = 1.57 . . . I = 100 . . . A(100) = 3.0*Q
X=Y*Z is eliminated.
X = A(I,J)* PI
Some programs used for performance analysis often contain such unnecessary operations. When you try to measure the performance of such programs compiled with Visual Fortran, these programs may show unrealistically good performance results. Realistic results are possible only with program units using their results in output statements.
Register Usage
A large program usually has more data that would benefit from being held in
Page 41 of 58
registers than there are registers to hold the data. In such cases, Visual Fortran typically tries to use the registers according to the following descending priority list: 1. 2. 3. 4. For temporary operation results, including array indexes For variables For addresses of arrays (base address) All other usages
Visual Fortran uses heuristic algorithms and a modest amount of computation to attempt to determine an effective usage for the registers. Holding Variables in Registers Because operations using registers are much faster than using memory, Visual Fortran generates code that uses the integer and floating-point registers instead of memory locations. Knowing when Visual Fortran uses registers may be helpful when doing certain forms of debugging. Visual Fortran uses registers to hold the values of variables whenever the Fortran language does not require them to be held in memory, such as holding the values of temporary results of subexpressions, even if /optimize:0 (no optimization) was specified. Visual Fortran may hold the same variable in different registers at different points in the program:
V = . . . X = . . . V = . . . Y = 3.0*Q
SIN(Y)*V
PI*X
COS(Y)*V
Visual Fortran may choose one register to hold the first use of V and another register to hold the second. Both registers can be used for other purposes at points in between. There may be times when the value of the variable does not exist anywhere in the registers. If the value of V is never needed in memory, it might not ever be assigned. Visual Fortran uses registers to hold the values of I, J, and K (so long as there are no other optimization effects, such as loops involving the variables):
A(I) = B(J) + C(K)
Page 42 of 58
In this case, K is loaded into only one register, which is used to index all three arrays at the same time.
Add (+), subtract (-), and multiply (*) operations if either operand is REAL Divide (/) operations if the divisor is REAL
For example, if variable R is REAL and A and B are COMPLEX, no conversion occurs with the following:
COMPLEX A, B . . . B = A + R
Global Optimizations
To enable global optimizations, use /optimize:2 or a higher optimization level. Using /optimize:2 or higher also enables local optimizations (/optimize:1). Global optimizations include:
l l l
l l
Data-flow analysis Split lifetime analysis Strength reduction (replaces a CPU-intensive calculation with one that uses fewer CPU cycles) Code motion (also called code hoisting) Instruction scheduling
Data-flow and split lifetime analysis (global data analysis) traces the values of variables and whole arrays as they are created and used in different parts of a program unit. During this analysis, Visual Fortran assumes that any pair of array references to a given array might access the same memory location, unless constant subscripts are used in both cases. To eliminate unnecessary recomputations of invariant expressions in loops, Visual Fortran hoists them out of the loops so they execute only once.
Page 43 of 58
Global data analysis includes which data items are selected for analysis. Some data items are analyzed as a group and some are analyzed individually. Visual Fortran limits or may disqualify data items that participate in the following constructs, generally because it cannot fully trace their values. Data items in the following declarations can make global optimizations less effective:
l
VOLATILE declarations VOLATILE declarations are needed to use certain run-time features of the operating system. Declare a variable as VOLATILE if the variable can be accessed using rules in addition to those provided by the Fortran 95/90 language. Examples include:
COMMON data items or entire common blocks that can change value by means other than direct assignment or during a routine call. For such applications, you must declare the variable or the COMMON block to which it belongs as volatile. An address not saved by the %LOC built-in function. Variables read or written by a signal handler, including those in a common block or module.
As requested by the VOLATILE statement, Visual Fortran disqualifies any volatile variables from global data analysis.
l
Subroutine calls or external function references Visual Fortran cannot trace data flow in a called routine that is not part of the program unit being compiled, unless the same DF command compiled multiple program units (see Compile With Appropriate Options and Multiple Source Files). Arguments passed to a called routine that are used again in a calling program are assumed to be modified, unless the proper INTENT is specified in an interface block (the compiler must assume they are referenced by the called routine).
Common blocks Visual Fortran limits optimizations on data items in common blocks. If common block data items are referenced inside called routines, their values might be altered. In the following example, variable I might be altered by FOO, so Visual Fortran cannot predict its value in subsequent references.
COMMON /X/ I DO J=1,N I = J
Page 44 of 58
Variables in Fortran 90 modules Visual Fortran limits optimizations on variables in Fortran 90 modules. Like common blocks, if the variables in Fortran 90 modules are referenced inside called routines, their values might be altered.
Variables referenced by a %LOC built-in function or variables with the TARGET attribute Visual Fortran limits optimizations on variables indirectly referenced by a %LOC function or on variables with the TARGET attribute, because the called routine may dereference a pointer to such a variable.
Equivalence groups An equivalence group is formed explicitly with the EQUIVALENCE statement or implicitly by the COMMON statement. A program section is a particular common block or local data area for a particular routine. Visual Fortran combines equivalence groups within the same program section and in the same program unit. The equivalence groups in separate program sections are analyzed separately, but the data items within each group are not, so some optimizations are limited to the data within each group.
Loop unrolling, including instruction scheduling (see Loop Unrolling) Code replication to eliminate branches (see Code Replication to Eliminate Branches) Padding the size of certain power-of-two arrays to allow more efficient cache use (see Use Arrays Efficiently)
Loop Unrolling
At optimization level /optimize:3 or above, Visual Fortran attempts to unroll
Page 45 of 58
certain innermost loops, minimizing the number of branches and grouping more instructions together to allow efficient overlapped instruction execution (instruction pipelining). The best candidates for loop unrolling are innermost loops with limited control flow. As more loops are unrolled, the average size of basic blocks increases. Loop unrolling generates multiple copies of the code for the loop body (loop code iterations) in a manner that allows efficient instruction pipelining. The loop body is replicated some number of times, substituting index expressions. An initialization loop might be created to align the first reference with the main series of loops. A remainder loop might be created for leftover work. The number of times a loop is unrolled can be determined either by the optimizer or by using the /unroll option, which can specify the limit for loop unrolling. Unless the user specifies a value, the optimizer optimizer will choose an unroll amount that minimizes the overhead of prefetching while also limiting code size expansion. Array operations are often represented as a nested series of loops when expanded into instructions. The innermost loop for the array operation is the best candidate for loop unrolling (like DO loops). For example, the following array operation (once optimized) is represented by nested loops, where the innermost loop is a candidate for loop unrolling:
A(1:100,2:30) = B(1:100,1:29) * 2.0
Page 46 of 58
Unoptimized Instructions
. . . branch to exit1 . . . branch to exit1 . . . exit1: move 1 into R0 return
Similarly, code replication can also occur within a loop that contains a small amount of shared code at the bottom of a loop and a case-type dispatch within the loop. The loop-end test-and-branch code might be replicated at the end of each case to create efficient instruction pipelining within the code for each case.
Automatic Inlining
To enable optimizations that perform automatic inlining, use /optimize:4 (or /optimize:5). Using /optimize:4 also enables local optimizations (/optimize:1), global optimizations (/optimize:2), and additional global optimizations (/optimize:3). To request inlining at lower optimization levels (/optimize:1, /optimize:2, or /optimize:3), use the /inline option. The default is /optimize:4 (unless /debug is specified), except in the visual development environment for a debug configuration.
Interprocedure Analysis
Compiling multiple source files at optimization level /optimize:4 or higher lets the compiler examine more code for possible optimizations, including multiple program units. This results in:
l l l
Inlining more procedures More complete global data analysis Reducing the number of external references to be resolved during linking
As more procedures are inlined, the size of the executable program and compile times may increase, but execution time should decrease.
Page 47 of 58
Inlining Procedures
Inlining refers to replacing a subprogram reference (such as a CALL statement or function invocation) with the replicated code of the subprogram. As more procedures are inlined, global optimizations often become more effective. The optimizer inlines small procedures, limiting inlining candidates based on such criteria as:
l l l
One of the /optimize options to control the optimization level. For example, specifying /optimize:4 or /optimize:5 enables interprocedure optimizations. Different /optimize options set different /inline:keyword options. For example, /optimize:4 sets /inline:speed.
One of the /inline options to directly control the inlining of procedures (see Controlling the Inlining of Procedures). For example, /inline:speed inlines more procedures than /inline:size. Certain /inline keywords require /optimize:1 or higher.
Page 48 of 58
l l
On loop transformations, see Loop Transformations. On software pipelining, see Software Pipelining (ia64 only).
Loop Transformations
The loop transformation optimizations are enabled by using the /transform_loops option or the /optimize:5 option. Loop transformation attempts to improve performance by rewriting loops to make better use of the memory system. By rewriting loops, the loop transformation optimizations can increase the number of instructions executed, which can degrade the run-time performance of some programs. To request loop transformation optimizations without software pipelining, do one of the following:
l l
Specify /optimize:5 with /nopipeline. Specify /transform_loops with /optimize:4, /optimize:3, or /optimize:2. This optimization is not performed at optimization levels below /optimize:2.
The loop transformation optimizations apply to array references within loops. These optimizations can improve the performance of the memory system and usually apply to multiple nested loops. The loops chosen for loop transformation optimizations are always counted loops. Counted loops are those loops that use a variable to count iterations in a manner that the number of iterations can be determined before entering the loop. For example, most DO loops are counted loops. Conditions that typically prevent the loop transformation optimizations from occurring include subprogram references that are not inlined (such as an external function call), complicated exit conditions, and uncounted loops. The types of optimizations associated with /transform_loops include the following:
l
Loop blocking Can minimize memory system use with multidimensional array elements by completing as many operations as possible on array elements currently in the cache. Also known as loop tiling.
Loop distribution Moves instructions from one loop into separate, new loops. This can reduce the amount of memory used during one loop so that the remaining memory may fit in the cache. It can also create improved opportunities for loop blocking.
Page 49 of 58
Loop fusion Combines instructions from two or more adjacent loops that use some of the same memory locations into a single loop. This can avoid the need to load those memory locations into the cache multiple times and improves opportunities for instruction scheduling.
Loop interchange Changes the nesting order of some or all loops. This can minimize the stride of array element access during loop execution and reduce the number of memory accesses needed. Also known as loop permutation.
Scalar replacement Replaces the use of an array element with a scalar variable under certain conditions.
Outer loop unrolling Unrolls the outer loop inside the inner loop under certain conditions to minimize the number of instructions and memory accesses needed. This also improves opportunities for instruction scheduling and scalar replacement.
For more information: On the interaction of compiler options and timing programs compiled with the loop transformation optimizations, see /[no]transform_loops.
Page 50 of 58
For instance, if software dependence analysis of data flow reveals that certain calculations can be done before or after that iteration of the loop, software pipelining reschedules those instructions ahead of or behind that loop iteration, at places where their execution can prevent instruction stalls or otherwise improve performance. Software pipelining also enables the prefetching of data to reduce the impact of cache misses. Software pipelining can be more effective when you combine /pipeline (or /optimize:5) with the appropriate /tune keyword for the target processor generation (see Requesting Optimized Code for a Specific Processor Generation). To specify software pipelining without loop transformation optimizations, do one of the following:
l
Specify /optimize:5 with /notransform_loops. Specify /pipeline with /optimize:4, /optimize:3, or /optimize:2. This optimization is not performed at optimization levels below /optimize:2.
For this version of Visual Fortran, loops chosen for software pipelining:
l
Are always innermost loops (those executed the most). Do not contain branches or procedure calls. Do not use COMPLEX floating-point data.
By modifying the unrolled loop and inserting instructions as needed before and/or after the unrolled loop, software pipelining generally improves run-time performance, except where the loops contain a large number of instructions with many existing overlapped operations. In this case, software pipelining may not have enough registers available to effectively improve execution performance. Run-time performance using /optimize:5 (or /pipeline) may not improve performance, as compared to using /optimize:4. For programs that contain loops that exhaust available registers, longer execution times may result with /optimize:5 or /pipeline. In cases where performance does not improve, consider compiling with the /unroll:1 option along with /optimize:5 or /pipeline, to possibly improve the effects of software pipelining. For more information:
Page 51 of 58
On the interaction of command-line options and timing programs compiled with software pipelining, see /[no]pipeline.
Options Set by the /fast Option Controlling Loop Unrolling Controlling the Inlining of Procedures Arithmetic Reordering Optimizations Dummy Aliasing Assumption Requesting Optimized Code for a Specific Processor Generation Requesting Code Generation for a Specific Processor Generation Loop Transformation Software Pipelining (ia64 only)
l l
/align:(dcommons,records,sequence) (see Data Alignment Considerations) /assume:noaccuracy_sensitive (see Arithmetic Reordering Optimizations) /architecture:host (see Requesting Code Generation for a Specific Processor Generation) /math_library:fast (see /math_library) /tune:host (see Requesting Optimized Code for a Specific Processor Generation)
Page 52 of 58
/inline:none (same as /noinline) Inlines statement functions but not other procedures. This type of inlining occurs if you specify /optimize:0 or /optimize:1 and omit /inline options.
/inline:manual Inlines statement functions but not other procedures. This type of inlining occurs if you specify /optimize:2 or /optimize:3 and omit /inline options.
/inline:size In addition to inlining statement functions, inlines any procedures that the Visual Fortran optimizer expects will improve run-time performance with no likely significant increase in program size.
/inline:speed In addition to inlining statement functions, inlines any procedures that the Visual Fortran optimizer expects will improve run-time performance with a likely significant increase in program size. This type of inlining occurs if you specify /optimize:4 or /optimize:5 and omit /inline options.
/inline:all Inlines every call that can possibly be inlined while generating correct code, including the following:
Statement functions (always inlined) Any procedures that Visual Fortran expects will improve run-time performance with a likely significant increase in program size. Any other procedures that can possibly be inlined and generate correct code. Certain recursive routines are not inlined to prevent
Page 53 of 58
For information on the inlining of other procedures (inlined at optimization level /optimize:4 or higher), see Inlining Procedures. Maximizing the types of procedures that are inlined usually improves run-time performance, but compile-time memory usage and the size of the executable program may increase. To determine whether using /inline:all benefits your particular program, time program execution for the same program compiled with and without /inline:all.
The results can be slightly different from the default /assume:accuracy_sensitive because of the way intermediate results are rounded. However, the /assume:noaccuracy_sensitive results are not categorically less accurate than those gained by the default. In fact, dot-product summations using /assume:noaccuracy_sensitive can produce more accurate results than those using /assume:accuracy_sensitive. The effect of /assume:noaccuracy_sensitive is important when Compaq Fortran hoists divide operations out of a loop. If /assume:noaccuracy_sensitive is in effect, the unoptimized loop becomes the optimized loop: Unoptimized Code
DO I=1,N . . . B(I)= A(I)/V END DO
Optimized Code
T= 1/V DO I=1,N . . . B(I)= A(I)*T END DO
The transformation in the optimized loop increases performance significantly, and loses little or no accuracy. However, it does have the potential for raising overflow or underflow arithmetic exceptions.
Page 54 of 58
C C 20
Page 55 of 58
The second DO loop contains assignments to DY. If DY is overlapped with DA, any of the assignments to DY might give DA a new value, and this overlap would affect the results. If this overlap is desired, then DA must be fetched from memory each time it is referenced. The repetitious fetching of DA degrades performance. Linking Routines with Opposite Settings You can link routines compiled with the /assume:dummy_aliases option to routines compiled with /assume:nodummy_aliases. For example, if only one routine is called with dummy aliases, you can use /assume:dummy_aliases when compiling that routine, and compile all the other routines with /assume:nodummy_aliases to gain the performance value of that option. Programs calling DAXPY with DA overlapping DY do not conform to the FORTRAN 77 and Fortran 90 standards. However, they are accommodated if /assume:dummy_aliases was used to compile the DAXPY routine.
Performance: Making Programs Run Faster generation, see the /architecture option.
Page 56 of 58
On Pentium 4 processor systems only, by generating code using the base set of Pentium instructions plus MMX, SSE, and SSE2 instruction extensions. On AMD Athlon processor systems only, by generating code using the base set of Pentium instructions plus MMX, SSE, 3DNow, and Enhanced 3DNow instruction extensions. On AMD K6_2, AMD K6_III, and AMD Athlon processor systems only, by generating code using the base set of Pentium instructions plus MMX, SSE, and 3DNow instruction extensions. On Intel Pentium III and Pentium 4 processor systems, and AMD K6_2, AMD K6_III, and AMD Athlon processor systems only, by generating code using the base set of Pentium instructions plus MMX and SSE instruction extensions. On Intel Pentium Pro, Pentium II, and higher Intel processor systems, and AMD K6 and higher AMD processor systems only, by generating code using the base set of Pentium instructions plus MMX instruction extensions. On Pentium (586) and higher Intel and AMD systems only, by generating code using the base set of Pentium instructions. On all ia32 Intel and AMD systems, by generating code using a generic blend of instructions that runs with moderate efficiency on all Intel and AMD systems.
You can also request that the type of generated code be determined by the host system being used to compile the application. For an ia64 system, request either the type of host system being used for compilation or request generic blend of instructions. The /arch:keyword option uses the same keywords as the /tune:keyword option. For more information:
l
See /architecture.
Page 57 of 58
Certain compiler source directives (cDEC$ prefix) can be used in place of some performance-related compiler options and provide more control of certain optimizations, as discussed in the following sections:
l l l
Using the cDEC$ OPTIONS Directive Using the cDEC$ UNROLL Directive to Control Loop Unrolling Using the cDEC$ IVDEP Directive to Control Certain Loop Optimizations
Page 58 of 58
The cDEC$ IVDEP directive tells the optimizer to begin dependence analysis by assuming all dependences occur in the same forward direction as their appearance in the normal scalar execution order. This contrasts with normal compiler behavior, which is for the dependence analysis to make no initial assumptions about the direction of a dependence. For more information:
l
Using QuickWin
Page 1 of 40
Using QuickWin
This chapter introduces the major categories of QuickWin library routines. It gives an overview of QuickWin features and their use in creating and displaying graphics, and customizing your QuickWin applications with custom menus and mouse routines. Drawing Graphics Elements, and Using Fonts from the Graphics Library cover graphics and fonts in more detail. The Visual Fortran QuickWin run-time library helps you turn graphics programs into simple Windows applications. Though the full capability of Windows is not available through QuickWin, QuickWin is simpler to learn and to use. QuickWin applications do support pixel-based graphics, real-coordinate graphics, text windows, character fonts, user-defined menus, mouse events, and editing (select/copy/paste) of text, graphics, or both. In Visual Fortran, graphics programs must be either Fortran QuickWin, Fortran Standard Graphics, Fortran Windows, or use OpenGL routines. Fortran Standard Graphics Applications are a subset of QuickWin that support only one window. You can choose the Fortran QuickWin or Standard Graphics application type from the drop-down list of available project types when you create a new project in the visual development environment. Or you can use the /libs:qwin compiler option for Fortran QuickWin or the /libs:qwins compiler option for Fortran Standard Graphics. Note that Fortran QuickWin and Standard Graphics applications cannot be DLLs, and QuickWin and Standard Graphics cannot be linked with run-time routines that are in DLLs. This means that the /libs:qwin option and the /libs:dll with /threads options cannot be used together. You can access the QuickWin routines library from Visual Fortran as well as other languages that support the Fortran calling conventions. The graphics package supports all video modes supported by Windows 2000, Windows NT 4, Windows Me, Windows 98, and Windows 95. A program using the QuickWin routines must explicitly access the QuickWin graphics library routines with the statement USE DFLIB (see USE Statement Needed for Fortran QuickWin Applications). This section includes the following topics:
l l l l
Capabilities of QuickWin Comparing QuickWin with Windows-Based Applications Using Win32 with QuickWin Types of QuickWin Programs
Using QuickWin
l l l l l l l l l l l
Page 2 of 40
The QuickWin User Interface USE Statement Needed for Fortran QuickWin Applications Creating QuickWin Windows Using Graphics and Character-Font Routines Defining Graphics Characteristics Displaying Graphics Output Working with Screen Images Enhancing QuickWin Applications Customizing QuickWin Applications QuickWin Programming Precautions Simulating Nonblocking I/O
Capabilities of QuickWin
You can use the QuickWin library to do the following:
l l
l l l l l l l l
Compile console programs into simple applications for Windows. Minimize and maximize QuickWin applications like any Windows-based application. Call graphics routines. Load and save bitmaps. Select, copy and paste text, graphics, or a mix of both. Detect and respond to mouse clicks. Display graphics output. Alter the default application menus or add programmable menus. Create custom icons. Open multiple child windows.
Your application has an OLE (Object Linking and Embedding) container. You want direct access to GDI (Graphical Data Interface) functions. You want to add your own customized Help information to QuickWin Help. You want to create something other than a standard SDI (Single Document Interface) or MDI (Multiple Document Interface) application. (For example, if you want your application to have a dialog such as Windows' Calculator in the client area.) You want to use a modeless dialog box rather than a modal dialog box.
Using QuickWin
Page 3 of 40
To build a Fortran Standard Graphics application from the command line, use the /libs:qwins option. For example:
DF /libs=qwins stdg_app.f90
For information about building projects in the visual development environment, see Building Programs and Libraries.
Using QuickWin
Page 4 of 40
Some of the QuickWin project Visual Fortran Samples are in folders in ...DF98 \SAMPLES\QUICKWIN. The following sections discuss the two types of QuickWin applications:
l l
Using QuickWin
Page 5 of 40
Applications. Using these routines to customize your QuickWin application is described in Customizing QuickWin Applications. MTRX.FOR Compiled as a QuickWin Application
Some of the Fortran QuickWin applications provided as Samples include PLATFORM and POLYDRAW (see the respective folders in ...\DF98 \SAMPLES\ADVANCED\WIN32).
l l
Window contents can be copied as bitmaps or text to the Clipboard for printing or pasting to other applications. In Fortran QuickWin applications, any portion of the window can be selected and copied. Vertical and horizontal scroll bars appear automatically, if needed. The base name of the application's .EXE file appears in the window's title bar. Closing the application window terminates the program.
In addition, the Fortran QuickWin application has a status bar and menu bar. The status bar at the bottom of the window reports the current status of the window program (for example, running or input pending). Default QuickWin Menus shows the default QuickWin menus.
Using QuickWin
Page 6 of 40
The default MDI (Multiple Document Interface) menu bar has six menus: File, Edit, View, State, Window, and Help. File Menu
Edit Menu
For instructions on using the Edit options within QuickWin see Editing Text and Graphics from the QuickWin Edit Menu. View Menu
The resulting graphics might appear somewhat distorted whenever the logical graphics screen is enlarged or reduced with the Size to Fit and Full Screen commands. While in Full Screen or Size To Fit mode, cursors are not scaled. State Menu
Window Menu
Using QuickWin
Page 7 of 40
Help Menu
For instructions on replacing the About default information within the Help menu with your own text message, see Defining an About Box. For instructions on how to create custom QuickWin menus, see Customizing QuickWin Applications.
Using QuickWin
l l l
Page 8 of 40
Giving a Window Focus and Setting the Active Window Keeping Child Windows Open Controlling Size and Position of Windows
If you use SETWINDOWCONFIG to set the variables in windowconfig to -1, the highest resolution will be set for your system, given the other fields you specify, if any. You can set the actual size of the window by specifying parameters that influence the window size -- the number of x and y pixels, the number of rows and columns, and the font size. If you do not call SETWINDOWCONFIG, the window defaults to the best possible resolution and a font size of 8 by 16. The number of colors depends on the video driver used. The font size, x pixels, y pixels, and columns and rows are related and cannot all
Using QuickWin
Page 9 of 40
be set arbitrarily. The following example specifies the number of x and y pixels and the font size and accepts the system calculation for the best number of rows and columns for the window:
USE DFLIB TYPE (windowconfig) wc LOGICAL status ! Set the x & y pixels to 800X600 and font size to 8x12. wc%numxpixels = 800 ! pixels on x-axis, window width wc%numypixels = 600 ! pixels on y-axis, window height wc%numtextcols = -1 ! -1 requests system default/calculation wc%numtextrows = -1 wc%numcolors = -1 wc%title = " "C wc%fontsize = #0008000C ! Request 8x12 pixel fonts status = SETWINDOWCONFIG(wc)
In this example:
l
The variables wc%numxpixels and wc%numypixels specify the size of the window, in this case 800 by 600 pixels. Within this window size, you can choose to specify either the font size (wc%fontsize) or the number of text columns (wc%numtextcols) and rows (wc%numtextrows). This example specifies the window size and font size, and lets the system calculate the number of text columns and rows. If you choose to specify the number of text columns and rows, you can let the system calculate (specify -1) either the font size or the window size.
The variable wc%fontsize is given as hexadecimal constant of #0008000C, which is interpreted in two parts: The left side of 0008 (8) specifies the width of the font, in pixels. The right side of 000C (12 in decimal) specifies the height of the font, in pixels. The variable wc%numtextrows is -1 and wc%numtextcols is -1, which allows the system to calculate the best available number of text columns and text rows to be used, as follows: The number of text columns is calculated as wc%numypixels (800) divided by the width of the font 8 (decimal) or 100. The number of text rows is calculated as wc%numxpixels (600) divided by the width of the font, 12 (decimal) or 50.
The requested font size is matched to the nearest available font size. If the matched size differs from the requested size, the matched size is used to determine the number of columns and rows. If scroll bars are needed (virtual window size exceeds the visible window size),
Using QuickWin
Page 10 of 40
because of the size required by horizontal and vertical scroll bars for a window, you may need to set the number of lines and columns to a value 1 or 2 greater than the number of rows and columns needed to display the application's information. If the requested configuration cannot be set, SETWINDOWCONFIG returns .FALSE. and calculates parameter values that will work and best fit the requested configuration. Another call to SETWINDOWCONFIG establishes these values:
IF(.NOT.status) status = SETWINDOWCONFIG(wc)
For an example, see the QuickWin Sample Cleanwin. For information on setting the graphics mode with SETWINDOWCONFIG, see Setting the Graphics Mode. Routines such as SETWINDOWCONFIG work on the window that is currently in focus. You can have multiple windows open as your application requires, but you need to decide which one gains focus. There is a single frame window and one or more child windows. A window is in focus right after it is opened, after I/O to it, and when FOCUSQQ is used. Clicking the mouse when the cursor is in a window will also bring the window into focus. For example, to set the characteristics for the window associated with unit 10, either gain focus with either an OPEN, a subsequent READ or WRITE statement to unit 10, or FOCUSQQ. For example, use OPEN:
open(unit=10, file='user') result = setwindowconfig(wc)
After you open unit 10, focus can be regained by a READ or WRITE statement to that unit. For example:
write(10,*) "Hello, this is unit 10"
For more information about when a window gains focus, see Giving a Window Focus and Setting the Active Window.
Page 11 of 40
Running a QuickWin application displays the frame window, but not the child window. You must call SETWINDOWCONFIG or execute an I/O statement or a graphics statement to display the child window. The window receives output by its unit number, as in:
OPEN (UNIT= 12, FILE= 'USER', TITLE= 'Product Matrix') WRITE (12, *) 'Enter matrix type: '
Child windows opened with FILE='USER' must be opened as sequential-access formatted files (the default). Other file specifications (direct-access, binary, or unformatted) result in run-time errors. The following example creates three child windows. A frame window is automatically created. Text is written to each so the child windows are visible:
program testch use dflib open(11,file="user") write(11,*) "Hello 11" open(12,file="user") write(12,*) "Hello 12" open(13,file="user") write(13,*) "Hello 13" write(13,*) "Windows 11, 12, and 13 can be read and written with normal" write(13,*) "Fortran I/O statements. The size of each window on the screen" write(13,*) "can be modified by SETWSIZEQQ. The size of the virtual window" write(13,*) "(i.e., a data buffer) can be modified by SETWINDOWCONFIG." read(13,*) end
Using QuickWin
Page 12 of 40
Using QuickWin
Page 13 of 40
the focus when a I/O statement is executed on that unit. For example:
OPEN (UNIT = 10, FILE = 'USER', IOFOCUS = .TRUE.)
With an explicit OPEN with FILE='USER', IOFOCUS defaults to .TRUE. For child windows opened implicitly (no OPEN statement before the READ WRITE, or PRINT) as unit 0, 5, or 6, IOFOCUS defaults to .FALSE.. If IOFOCUS=.TRUE., the child window receives focus prior to each READ, WRITE, or PRINT. Calls to OUTTEXT or graphics functions (for example, OUTGTEXT, LINETO, and ELLIPSE) do not cause the focus to shift. If you use IOFOCUS with any unit other than a QuickWin child window, a run-time error occurs. The focus shifts to a window when it is given the focus with FOCUSQQ, when it is selected by a mouse click, or when an I/O operation other than a graphics operation is performed on it, unless the window was opened with IOFOCUS=.FALSE.. INQFOCUSQQ determines which unit has the focus. For example:
USE DFLIB INTEGER(4) status, focusunit OPEN(UNIT = 10, FILE = 'USER', TITLE = 'Child Window 1') OPEN(UNIT = 11, FILE = 'USER', TITLE = 'Child Window 2') ! Give focus to Child Window 2 by writing to it: WRITE (11, *) 'Giving focus to Child 2.' ! Give focus to Child Window 1 with the FOCUSQQ function: status = FOCUSQQ(10) ... ! Find out the unit number of the child window that currently has focus: status = INQFOCUSQQ(focusunit)
SETACTIVEQQ makes a child window active without bringing it to the foreground. GETACTIVEQQ returns the unit number of the currently active child window. GETHWNDQQ converts the unit number into a Windows handle for functions that require it.
Using QuickWin
Page 14 of 40
The options for the qwinfo type are listed under SETWSIZEQQ in the Language Reference. GETWSIZEQQ returns the position and the current or maximum window size of the current frame or child window. To access information about a child window, specify the unit number associated with it. Unit numbers 0, 5, and 6 refer to the default startup window if you have not explicitly opened them with the OPEN statement. To access information about the frame window, specify the unit number as the symbolic constant QWIN$FRAMEWINDOW. For example:
USE DFLIB INTEGER(4) status TYPE (QWINFO) winfo OPEN (4, FILE='USER') ... ! Get current size of child window associated with unit 4. status = GETWSIZEQQ(4, QWIN$SIZECURR, winfo) WRITE (*,*) "Child window size is ", winfo.H, " by ", winfo.W ! Get maximum size of frame window. status = GETWSIZEQQ(QWIN$FRAMEWINDOW, QWIN$SIZEMAX, winfo) WRITE (*,*) "Max frame window size is ", winfo.H, " by ", winfo.W
SETWSIZEQQ is used to set the visible window position and size. For example:
USE DFLIB INTEGER(4) status TYPE (QWINFO) winfo OPEN (4, FILE='USER') winfo.H = 30 winfo.W = 80 winfo.TYPE = QWIN$SET status = SETWSIZEQQ(4, winfo)
Using QuickWin
Page 15 of 40
Graphics routines are functions and subroutines that draw lines, rectangles, ellipses, and similar elements on the screen. Font routines create text in a variety of sizes and styles. The QuickWin graphics library provides routines that:
l l l l l l l l
Change the window's dimensions. Set coordinates. Set color palettes. Set line styles, fill masks, and other figure attributes. Draw graphics elements. Display text in several character styles. Display text in fonts compatible with Microsoft Windows. Store and retrieve screen images.
Selecting Display Options Setting Graphics Coordinates Using Color Setting Figure Properties
Page 16 of 40
l l
Fixed physical coordinates, which are determined by the hardware and the video mode used Viewport coordinates, which you can define in the application Window coordinates, which you can define to simplify scaling of floatingpoint data values
Unless you change it, the viewport-coordinate system is identical to the physical-coordinate system. The physical origin (0, 0) is always in the upper-left corner of the display. For QuickWin, display means a child window's client area, not the actual monitor screen (unless you go to Full Screen mode). The x-axis extends in the positive direction left to right, while the y-axis extends in the positive direction top to bottom. The default viewport has the dimensions of the selected mode. In a QuickWin application, you can draw outside of the child window's current client area. If you then make the child window bigger, you will see what was previously outside the frame. You can also use coordinate routines to convert between physical-, viewport-, and window-coordinate systems. (For more detailed information on coordinate systems, see Drawing Graphics Elements.) You can set the pixel dimensions of the x- and y-axes with SETWINDOWCONFIG. You can access these values through the wc%numxpixels and wc%numypixels values returned by GETWINDOWCONFIG. Similarly, GETWINDOWCONFIG also returns the range of colors available in the current mode through the wc%numcolors value. You can also define the graphics area with SETCLIPRGN and SETVIEWPORT. Both of these functions define a subset of the available window area for graphics output. SETCLIPRGN does not change the viewport coordinates, but merely masks part of the screen. SETVIEWPORT resets the viewport bounds to the limits you give it and sets the origin to the upper-left corner of this region. The origin of the viewport-coordinate system can be moved to a new position relative to the physical origin with SETVIEWORG. Regardless of the viewport coordinates, however, you can always locate the current graphics output position with GETCURRENTPOSITION and GETCURRENTPOSITION_W. (For more detailed information on viewports and clipping regions, see Drawing Graphics Elements.) Using the window-coordinate system, you can easily scale any set of data to fit on the screen. You define any range of coordinates (such as 0 to 5000) that works well for your data as the range for the window-coordinate axes. By telling the program that you want the window-coordinate system to fit in a particular
Using QuickWin
Page 17 of 40
area on the screen (map to a particular set of viewport coordinates), you can scale a chart or drawing to any size you want. SETWINDOW defines a windowcoordinate system bounded by the specified values. See the Visual Fortran Sample SINE.F90 in the ...\DF98\SAMPLES\TUTORIAL folder for an example of this technique. GETPHYSCOORD converts viewport coordinates to physical coordinates, and GETVIEWCOORD translates from physical coordinates to viewport coordinates. Similarly, GETVIEWCOORD_W converts window coordinates to viewport coordinates, and GETWINDOWCOORD converts viewport coordinates to window coordinates. For more information:
l
Using Color
If you have a VGA machine, you are restricted to displaying at most 256 colors at a time. These 256 colors are held in a palette. You can choose the palette colors from a range of 262,144 colors (256K), but only 256 at a time. The palette routines REMAPPALETTERGB and REMAPALLPALETTERGB assign RedGreen-Blue (RGB) colors to palette indexes. Functions and subroutines that use color indexes create graphic outputs that depend on the mapping between palette indexes and RGB colors. REMAPPALETTERGB remaps one color index to an RGB color, and REMAPALLPALETTERGB remaps the entire palette, up to 236 colors, (20 colors are reserved by the system). You cannot remap the palette on machines capable of displaying 20 colors or fewer. SVGA and true color video adapters are capable of displaying 262,144 (256K) colors and 16.7 million colors respectively. If you use a palette, you are restricted to the colors available in the palette. To access the entire set of available colors, not just the 256 or fewer colors in the palette, you should use functions that specify a color value directly. These functions end in RGB and use Red-Green-Blue color values, not indexes to a palette. For example, SETCOLORRGB, SETTEXTCOLORRGB, and SETPIXELRGB specify a direct color value, while SETCOLOR, SETTEXTCOLOR, and SETPIXEL each specify a palette color index. If you are displaying more than 256 colors simultaneously, you need to use the RGB direct color value functions exclusively. To set the physical display properties of your monitor, open the Control Panel and click the Display icon.
Using QuickWin
Page 18 of 40
QuickWin only supports a 256-color pallette, regardless of the number of colors set for the monitor. For more information on setting colors, see Adding Color in Drawing Graphics Elements.
Using QuickWin
Page 19 of 40
The run-time graphics library routines can draw geometric features, display text, display font-based characters, and transfer images between memory and the screen. These capabilities are discussed in the following topics:
l l l
Drawing Graphics
If you want anything other than the default line style (solid), mask (no mask), background color (black), or foreground color (white), you must call the appropriate routine before calling the drawing routine. Subsequent output routines employ the same attributes until you change them or open a new child window. The following is a list of routines that ask about the current graphics settings, set new graphics settings, and draw graphics: Routine ARC, ARC_W CLEARSCREEN ELLIPSE, ELLIPSE_W FLOODFILL, FLOODFILL_W Description Draws an arc Clears the screen, viewport, or text window Draws an ellipse or circle Fills an enclosed area of the screen with the current color index using the current fill mask Fills an enclosed area of the screen with the current RGB color using the current fill mask Determines the endpoints of the most recently drawn arc or pie Returns the coordinates of the current graphics-output position Returns a pixel's color index Returns a pixel's Red-Green-Blue color value
Using QuickWin GETPIXELS GETPIXELSRGB GRSTATUS INTEGERTORGB LINETO, LINETO_W LINETOAR, LINETOAREX MOVETO, MOVETO_W PIE, PIE_W POLYGON, POLYGON_W RECTANGLE, RECTANGLE_W RGBTOINTEGER
Page 20 of 40 Gets the color indices of multiple pixels Gets the Red-Green-Blue color values of multiple pixels Returns the status (success or failure) of the most recently called graphics routine Convert a true color value into its red, green, and blue components Draws a line from the current graphicsoutput position to a specified point Draws lines from arrays at x,y coordinate points Moves the current graphics-output position to a specified point Draws a pie-slice-shaped figure Draws a polygon Draws a rectangle Convert a trio of red, green, and blue values to a true color value for use with RGB functions and subroutines Sets a pixel at a specified location to a color index Sets a pixel at a specified location to a Red-Green-Blue color value Set the color indices of multiple pixels Set the Red-Green-Blue color value of multiple pixels
Most of these routines have multiple forms. Routine names that end with _W use the window-coordinate system and REAL(8) argument values. Routines without this suffix use the viewport-coordinate system and INTEGER(2) argument values. Curved figures, such as arcs and ellipses, are centered within a bounding rectangle, which is specified by the upper-left and lower-right corners of the
Using QuickWin
Page 21 of 40
rectangle. The center of the rectangle becomes the center for the figure, and the rectangle's borders determine the size of the figure. In the following figure, the points (x1, y1) and (x2, y2) define the bounding rectangle. Bounding Rectangle
Using QuickWin GETTEXTPOSITION GETTEXTWINDOW OUTTEXT SCROLLTEXTWINDOW SETBKCOLOR SETBKCOLORRGB SETTEXTCOLOR SETTEXTCOLORRGB SETTEXTPOSITION SETTEXTWINDOW WRAPON Returns the current text-output position
Page 22 of 40
Returns the boundaries of the current text window Sends text to the screen at the current position Scrolls the contents of a text window Sets the current background color index Sets the current background Red-Green-Blue color value Sets the current text color to a new color index Sets the current text color to a new Red-Green-Blue color value Changes the current text position Sets the current text-display window Turns line wrapping on or off
These routines do not provide text-formatting capabilities. If you want to print integer or floating-point values, you must convert the values into a string (using an internal WRITE statement) before calling these routines. The text routines specify all screen positions in character-row and column coordinates. SETTEXTWINDOW is the text equivalent of the SETVIEWPORT graphics routine, except that it restricts only the display area for text printed with OUTTEXT, PRINT, and WRITE. GETTEXTWINDOW returns the boundaries of the current text window set by SETTEXTWINDOW. SCROLLTEXTWINDOW scrolls the contents of a text window. OUTTEXT, PRINT, and WRITE display text strings written to the current text window. Warning: The WRITE statement sends its carriage return (CR) and line feed (LF) to the screen at the beginning of the first I/O statement following the WRITE statement. This can cause unpredictable text positioning if you mix the graphics routines SETTEXTPOSITION and OUTTEXT with the WRITE statement. To minimize this effect, use the backslash ( \ ) or dollar sign ( $ ) format descriptor (to suppress CR-LF) in the associated FORMAT statement. For more information:
l
Using QuickWin
l
Page 23 of 40
On allocating a console to display text in a QuickWin application, see Using the Console.
SETGTEXTROTATION
Characters may be drawn ("mapped") in one of two ways: as bitmapped letters (a "picture" of the letter) or as TrueType characters. See Using Fonts from the Graphics Library, for detailed explanations and examples of how to use the font routines from the QuickWin Library. For more information on these routines, see the Language Reference.
Using QuickWin
Page 24 of 40
Transfer images between memory buffers and the screen Transferring images from buffers is a quick and flexible way to move things around the screen. Memory images can interact with the current screen image; for example, you can perform a logical AND of a memory image and the current screen or superimpose a negative of the memory image on the screen.
Transfer images between the screen and Windows bitmap files Transferring images from files gives access to images created by other programs, and saves graphs and images for later use. However, images loaded from bitmap files overwrite the portion of the screen they are pasted into and retain the attributes they were created with, such as the color palette, rather than accepting current attributes.
Transfer images between the screen and the Clipboard from the QuickWin Edit menu Editing screen images from the QuickWin Edit menu is a quick and easy way to move and modify images interactively on the screen, retaining the current screen attributes, and also provides temporary storage (the Clipboard) for transferring images among applications.
These routines allow you to cut, paste, and move images around the screen.
Using QuickWin
Page 25 of 40
You can use a Windows format bitmap file created with a graphics program as a backdrop for graphics that you draw with the Visual Fortran graphics functions and subroutines. For more information:
l
Using QuickWin
Page 26 of 40
l l
If you have chosen the Select All option from the Edit menu, the whole screen is selected and you cannot then select a portion of the screen. Text selections are not bounded by the current text window set with SETTEXTWINDOW. When text is copied to the Clipboard, trailing blanks in a line are removed. Text that is written to a window can be overdrawn by graphics. In this case, the text is still present in the screen text buffer, though not visible on the screen. When you select a portion of the screen to copy, you can select text that is actually present but not visible, and that text will be copied onto the Clipboard. When you chose Select Text or Select Graphics from the Edit menu, the application is paused, a caret (^) appears at the top left corner of the currently active window, all user-defined callbacks are disabled, and the window title changes to "Mark Text - windownam "or "Mark Graphics windowname", where windowname is the name of the currently active window. As soon as you begin selection (by pressing an arrow key or a mouse button), the Window title changes to "Select Text - windowname" or "Select Graphics - windowname" and selection begins at that point. If you do not want selection to begin in the upper-left corner, your first action when "Mark Text" or "Mark Graphics" appears in the title is to use the mouse to place the cursor at the position where selection is to be begin.
Using QuickWin Menu items CLICKMENUQQ APPENDMENUQQ DELETEMENUQQ INSERTMENUQQ MODIFYMENUFLAGSQQ MODIFYMENUROUTINEQQ MODIFYMENUSTRINGQQ SETWINDOWMENUQQ
Page 27 of 40 Simulates the effect of clicking or selecting a menu item Appends a menu item Deletes a menu item Inserts a menu item Modifies a menu item's state Modifies a menu item's callback routine Changes a menu item's text string Sets the menu to which a list of current child window names are appended Enables (or disables) use of the arrow directional keys and page keys as input (see the Sample DIRKEYS.F90 in the ...\DF98 \SAMPLES\ADVANCED\DIRKEYS folder) Changes any QuickWin message, including status bar messages, state messages and dialog box messages Registers the application defined routines to be called on mouse events Removes the routine registered by REGISTERMOUSEEVENT Blocks return until a mouse event occurs
Directional keys
PASSDIRKEYSQQ
QuickWin messages
SETMESSAGEQQ
Mouse actions
REGISTERMOUSEEVENT
UNREGISTERMOUSEEVENT WAITONMOUSEEVENT
Page 28 of 40
The following topics describe how to customize and fine-tune your QuickWin applications:
l l l l l l
Program Control of Menus Changing Status Bar and State Messages Displaying Message Boxes Defining an About Box Using Custom Icons Using a Mouse
Controlling the Initial Menu and Frame Window Deleting, Inserting, and Appending Menu Items Modifying Menu Items Creating a Menu List of Available Child Windows Simulating Menu Selections
Controlling the Initial Menu and Frame Window You can change the initial appearance of an application's default frame window and menus by defining an INITIALSETTINGS function. If no user-defined INITIALSETTINGS function is supplied, QuickWin calls a predefined (default) INITIALSETTINGS routine that provides a default frame window and menu. Your application does not call INITIALSETTINGS. If you supply the function in your project, QuickWin calls it automatically. If you supply it, INITIALSETTINGS can call QuickWin functions that set the initial menus and the size and position of the frame window. Besides the menu functions, SETWSIZEQQ can be called from your INITIALSETTINGS function to adjust the frame window size and position before the window is first drawn. The following is a sample of INITIALSETTINGS:
LOGICAL(4) FUNCTION INITIALSETTINGS( ) USE DFLIB LOGICAL(4) result TYPE (qwinfo) qwi ! Set window frame size. qwi%x = 0 qwi%y = 0 qwi%w = 400
Using QuickWin
Page 29 of 40
qwi%h = 400 qwi%type = QWIN$SET i = SetWSizeQQ( QWIN$FRAMEWINDOW, qwi ) ! Create first menu called Games. result = APPENDMENUQQ(1, $MENUENABLED, '&Games'C, NUL ) ! Add item called TicTacToe. result = APPENDMENUQQ(1, $MENUENABLED, '&TicTacToe'C, WINPRINT) ! Draw a separator bar. result = APPENDMENUQQ(1, $MENUSEPARATOR, ''C, NUL ) ! Add item called Exit. result = APPENDMENUQQ(1, $MENUENABLED, 'E&xit'C, WINEXIT ) ! Add second menu called Help. result = APPENDMENUQQ(2, $MENUENABLED, '&Help'C, NUL ) result = APPENDMENUQQ(2, $MENUENABLED, '&QuickWin Help'C, WININDEX) INITIALSETTINGS= .true. END FUNCTION INITIALSETTINGS
QuickWin executes your INITIALSETTINGS function during initialization, before creating the frame window. When your function is done, control returns to QuickWin and it does the remaining initialization. The control then passes to the Visual Fortran application. You only need to include the code for an INITIALSETTINGS function in your project. If it is part of your project, you do not need to call your INITIALSETTINGS function. The INITIALSETTINGS function can specify the position and size of the frame window and the contents of the menus. Because the INITIALSETTINGS function is executed before creating the frame window, it must not call any routines that require that frame window initialization be complete. For example, routines that refer to the child window in focus (such as SETWINDOWCINFIG) or a specific child window unit number (such as OPEN) should not be called from the INITIALSETTINGS function. Your INITIALSETTINGS function should return .TRUE. if it succeeds, and .FALSE. otherwise. The QuickWin default function returns a value of .TRUE. only. Note that default menus are created after INITIALSETTINGS has been called, and only if you do not create your own menus. Therefore, using DELETEMENUQQ, INSERTMENUQQ, APPENDMENUQQ, and the other menu configuration QuickWin functions while in INITIALSETTINGS affects your custom menus, not the default QuickWin menus. Deleting, Inserting, and Appending Menu Items Menus are defined from left to right, starting with 1 at the far left. Menu items are defined from top to bottom, starting with 0 at the top (the menu title itself). Within INITIALSETTINGS, if you supply it, you can delete, insert, and append menu items in custom menus. Outside INITIALSETTINGS, you can alter the default QuickWin menus as well as custom menus at any point in your
Using QuickWin
Page 30 of 40
application. (Default QuickWin menus are not created until after INITIALSETTINGS has run and only if you do not create custom menus.) To delete a menu item, specify the menu number and item number in DELETEMENUQQ. To delete an entire menu, delete item 0 of that menu. For example:
USE DFLIB LOGICAL status status = DELETEMENUQQ(1, 2) ! ! status = DELETEMENUQQ(5, 0) ! !
Delete the second menu item from menu 1 (the default FILE menu). Delete menu 5 (the default Windows menu).
INSERTMENUQQ inserts a menu item or menu and registers its callback routine. QuickWin supplies several standard callback routines such as WINEXIT to terminate a program, WININDEX to list QuickWin Help, and WINCOPY which copies the contents of the current window to the Clipboard. A list of available callbacks is given in the Language Reference for INSERTMENUQQ and APPENDMENUQQ. Often, you will supply your own callback routines to perform a particular action when a user selects something from one of your menus. In general, you should not assign the same callback routine to more than one menu item because a menu item's state might not be properly updated when you change it (put a check mark next to it, gray it out, or disable, or enable it). You cannot insert a menu item or menu beyond the existing number; for example, inserting item 7 when 5 and 6 have not been defined yet. To insert an entire menu, specify menu item 0. The new menu can take any position among or immediately after existing menus. If you specify a menu position occupied by an existing menu, the existing menu and any menus to the right of the one you add are shifted right and their menu numbers are incremented. For example, the following code inserts a fifth menu item called Position into menu 5 (the default Windows menu):
USE DFLIB LOGICAL(4) status status = INSERTMENUQQ (5, 5, $MENUCHECKED, 'Position'C, WINPRINT)
The next code example inserts a new menu called My List into menu position 3. The menu currently in position 3 and any menus to the right (the default menus View, State, Windows, and Help) are shifted right one position:
USE DFLIB LOGICAL(4) status status = INSERTMENUQQ(3,0, $MENUENABLED, 'My List'C, WINSTATE)
Using QuickWin
Page 31 of 40
You can append a menu item with APPENDMENUQQ. The item is added to the bottom of the menu list. If there is no item yet for the menu, your appended item is treated as the top-level menu item, and the string you assign to it appears on the menu bar. The QuickWin menu routines like INSERTMENUQQ and APPENDMENUQQ let you to create a single level of menu items beneath a menu name. You cannot create submenus with the QuickWin project type. The following code uses APPENDMENUQQ to append the menu item called Cascade Windows to the first menu (the default File menu):
USE DFLIB LOGICAL(4) status status = APPENDMENUQQ(1, $MENUCHECKED, 'Cascade Windows'C, & WINCASCADE)
&
The $MENUCHECKED flag in the example puts a check mark next to the menu item. To remove the check mark, you can set the flag to $MENUUNCHECKED in the MODIFYMENUFLAGSQQ function. Some predefined routines (such as WINSTATUS) take care of updating their own check marks. However, if the routine is registered to more than one menu item, the check marks might not be properly updated. See APPENDMENUQQ or INSERTMENUQQ in the Language Reference for the list of callback routines and other flags. Modifying Menu Items MODIFYMENUSTRINGQQ can modify the string identifier of a menu item, MODIFYMENUROUTINEQQ can modify the callback routine called when the item is selected, and MODIFYMENUFLAGSQQ can modify a menu item's state (such as enabled, grayed out, checked, and so on). The following example code uses MODIFYMENUSTRINGQQ to modify the menu string for the fourth item in the first menu (the File menu by default) to Tile Windows, it uses MODIFYMENUROUTINEQQ to change the callback routine called if the item is selected to WINTILE, and uses MODIFYMENUFLAGSQQ to put a check mark next to the menu item:
status = MODIFYMENUSTRINGQQ( 1, 4, 'Tile Windows'C) status = MODIFYMENUROUTINEQQ( 1, 4, WINTILE) status = MODIFYMENUFLAGSQQ( 1, 4, $MENUCHECKED)
Creating a Menu List of Available Child Windows By default, the Windows menu contains a list of all open child windows in your QuickWin applications. SETWINDOWMENUQQ changes the menu which lists the currently open child windows to the menu you specify. The list of child window names is appended to the end of the menu you choose and deleted from any
Using QuickWin other menu that previously contained it. For example:
Page 32 of 40
USE DFLIB LOGICAL(4) status ... ! Append list of open child windows to menu 1 (the default File menu) status = SETWINDOWMENUQQ(1)
Simulating Menu Selections CLICKMENUQQ simulates the effect of clicking or selecting a menu command from the Window menu. The QuickWin application behaves as though the user had clicked or selected the command. The following code fragment simulates the effect of selecting the Tile item from the Window menu:
USE DFLIB INTEGER(4) status status = CLICKMENUQQ(QWIN$TILE)
Items from the Window menu can be specified in CLICKMENUQQ. Other predefined routines such as WINFULLSCREEN and WINSIZETOFIT (see the callback subroutine names listed in APPENDMENUQQ) can be invoked by calling CLICKMENUQQ with an argument containing the LOC intrinsic function before the callback routine name. For example, the following program calls WINSIZETOFIT:
! Some of the callback subroutine names listed in APPENDMENUQQ may not ! work or be useful in this context, but they will be "called". ! Run the program and note how the scroll bars disappear use dflib integer*4 result character*1 ch print *,'type a character' read(*,*) ch result = clickmenuqq(loc(WINSIZETOFIT)) print *,'type a character' read(*,*) ch end
This function is useful for localizing your QuickWin applications for countries with different native languages. A list of message IDs is given in SETMESSAGEQQ in the Language Reference.
Using QuickWin
Page 33 of 40
&
Using QuickWin
Page 34 of 40
import it, not draw it, select Resource from the Insert menu, then select Import from the buttons in the Resource dialog. You will be prompted for the file containing your icon. 3. Name the icon. The frame window's icon must have the name "frameicon," and the child window's icon must have the name "childicon." These names must be entered as strings into the Icon Properties dialog box. To display the Icon Properties dialog box, double-click in the icon editor area outside the icon's grid or press ALT+ENTER. In the ID field on the General tab of Icon Properties dialog box, type over the default icon name with "frameicon" or "childicon." You must add the quotation marks to the text you type in order to make the name be interpreted as a string. Your icon will be saved in a file with the extension .ICO. 4. Create a script file to hold your icons. Select File/Save As. You will be prompted for the name of the script file that will contain your icons. Name the script file. It must end with the extension .RC; for example, myicons.rc. Using this method, the icons and their string values will be automatically saved in the script file. (Alternatively, you can create a script file with any editor and add the icon names and their string values by hand.) 5. Add the script file to the project that contains your QuickWin application. Select Build and the script file will be built into the application's executable. (The compiled script file will have the extension .RES.) When you run your application, the icon you created will take the place of the default child or frame icon. Your custom icon appears in the upper-left corner of the window frame. When you minimize the window, the icon appears on the left of the minimized window bar.
Using a Mouse
Your applications can detect and respond to mouse events, such as left mouse button down, right mouse button down, or double-click. Mouse events can be used as an alternative to keyboard input or for manipulating what is shown on the screen. QuickWin provides two types of mouse functions: 1. Event-based functions, which call an application-defined callback routine when a mouse click occurs 2. Blocking (sequential) functions, which provide blocking functions that halt an application until mouse input is made
Using QuickWin
Page 35 of 40
The mouse is an asynchronous device, so the user can click the mouse anytime while the application is running (mouse input does not have to be synchronized to anything). When a mouse-click occurs, Windows sends a message to the application, which takes the appropriate action. Mouse support in applications is most often event-based, that is, a mouse-click occurs and the application does something. However, an application can use blocking functions to wait for a mouse-click. This allows an application to execute in a particular sequential order and yet provide mouse support. QuickWin performs default processing based on mouse events. To change the shape of the mouse cursor, use the SETMOUSECURSOR routine. Event-Based Functions The QuickWin function REGISTERMOUSEEVENT registers the routine to be called when a particular mouse event occurs (left mouse button, right mouse button, double-click, and so on). You define what events you want it to handle and the routines to be called if those events occur. UNREGISTERMOUSEEVENT unregisters the routines so that QuickWin doesn't call them but uses default handling for the particular event. By default, QuickWin typically ignores events except when mouse-clicks occur on menus or dialog controls. Note that no events are received on a minimized window. A window must be restored or maximized in order for mouse events to happen within it. For example:
USE DFLIB INTEGER(4) result OPEN (4, FILE= 'USER') ... result = REGISTERMOUSEEVENT (4, MOUSE$LBUTTONDBLCLK, CALCULATE)
This registers the routine CALCULATE, to be called when the user double-clicks the left mouse button while the mouse cursor is in the child window opened as unit 4. The symbolic constants available to identify mouse events are: Mouse Event 1 MOUSE$LBUTTONDOWN MOUSE$LBUTTONUP MOUSE$LBUTTONDBLCLK Description Left mouse button down Left mouse button up Left mouse button double-click
Page 36 of 40 Right mouse button down Right mouse button up Right mouse button double-click Mouse moved
For every BUTTONDOWN and BUTTONDBLCLK event there is an associated BUTTONUP event. When the user double-clicks, four events happen: BUTTONDOWN and BUTTONUP for the first click, and BUTTONDBLCLK and BUTTONUP for the second click. The difference between getting BUTTONDBLCLK and BUTTONDOWN for the second click depends on whether the second click occurs in the double-click interval, set in the system's CONTROL PANEL/MOUSE. To unregister the routine in the preceding example, use the following code:
result = UNREGISTERMOUSEEVENT (4, MOUSE$LBUTTONDBLCLK)
If REGISTERMOUSEEVENT is called again without unregistering a previous call, it overrides the first call. A new callback routine is then called on the specified event. The callback routine you create to be called when a mouse event occurs should have the following prototype:
INTERFACE SUBROUTINE MouseCallBackRoutine (unit, mouseevent, keystate, & & MouseXpos,MouseYpos) INTEGER unit INTEGER mouseevent INTEGER keystate INTEGER MouseXpos INTEGER MouseYpos END SUBROUTINE END INTERFACE
The unit parameter is the unit number associated with the child window where events are to be detected, and the mouseevent parameter is one of those listed in the preceding table. The MouseXpos and the MouseYpos parameters specify the x and y positions of the mouse during the event. The keystate parameter indicates the state of the shift and control keys at the time of the mouse event, and can be any ORed combination of the following constants:
Using QuickWin
Page 37 of 40
Description Left mouse button down during event Right mouse button down during event Shift key held down during event Control key held down during event
QuickWin callback routines for mouse events should do a minimum of processing and then return. While processing a callback, the program will appear to be non-responsive because messages are not being serviced, so it is important to return quickly. If more processing time is needed in a callback, another thread should be started to perform this work; threads can be created by calling the Win32 API CreateThread. (For more information on creating and using threads, see Creating Multithread Applications.) If a callback routine does not start a new thread, the callback will not be re-entered until it is done processing. Note: In event-based functions, there is no buffering of events. Therefore, issues such as multithreading and synchronizing access to shared resources must be addressed. To avoid multithreading problems, use blocking functions rather than event-based functions. Blocking functions work well in applications that proceed sequentially. Applications where there is little sequential flow and the user jumps around the application are probably better implemented as event-based functions. Blocking (Sequential) Functions The QuickWin blocking function WAITONMOUSEEVENT blocks execution until a specific mouse input is received. This function is similar to INCHARQQ, except that it waits for a mouse event instead of a keystroke. For example:
USE DFLIB INTEGER(4) mouseevent, keystate, x, y, result ... mouseevent = MOUSE$RBUTTONDOWN .OR. MOUSE$LBUTTONDOWN result = WAITONMOUSEEVENT (mouseevent, keystate, x , y) ! Wait ! until right or left mouse button clicked, then check the keystate ! with the following: if ((MOUSE$KS_SHIFT .AND. keystate) == MOUSE$KS_SHIFT) then & & write (*,*) 'Shift key was down' if ((MOUSE$KS_CONTROL .AND. keystate) == MOUSE$KS_CONTROL) then & & write (*,*) 'Ctrl key was down'
Using QuickWin
Page 38 of 40
Your application passes a mouse event parameter, which can be any ORed combination of mouse events, to WAITONMOUSEEVENT. The function then waits and blocks execution until one of the specified events occurs. It returns the state of the Shift and Ctrl keys at the time of the event in the parameter keystate, and returns the position of the mouse when the event occurred in the parameters x and y. A mouse event must happen in the window that had focus when WAITONMOUSEEVENT was initially called. Mouse events in other windows will not end the wait. Mouse events in other windows cause callbacks to be called for the other windows, if callbacks were previously registered for those windows. Default QuickWin Processing QuickWin performs some actions based on mouse events. It uses mouse events to return from the FullScreen mode and to select text and/or graphics to copy to the Clipboard. Servicing the mouse event functions takes precedence over return from FullScreen mode. Servicing mouse event functions does not take precedence over Cut/Paste selection modes. Once selection mode is over, processing of mouse event functions resumes.
Blocking Procedures
Procedures that wait for an event before allowing the program to proceed, such as READ or WAITONMOUSEEVENT, both of which wait for user input, are called blocking procedures because they block execution of the program until the awaited event occurs. QuickWin child processes can contain multiple callback routines; for example, a different routine to be called for each menu selection and each kind of mouse-click (left button, right button, double-click, and so on). Problems can arise when a process and its callback routine, or two callback routines within the same process, both contain blocking procedures. This is because each QuickWin child process supports a primary and secondary thread. As a result of selecting a menu item, a menu procedure may call a blocking
Using QuickWin
Page 39 of 40
procedure, while the main thread of the process has also called a blocking procedure. For example, say you have created a file menu, which contains an option to LOAD a file. Selecting the LOAD menu option calls a blocking function that prompts for a filename and waits for the user to enter the name. However, a blocking call such as WAITONMOUSEEVENT can be pending in the main process thread when the user selects the LOAD menu option, so two blocking functions are initiated. When QuickWin has two blocking calls pending, it displays a message in the status bar that corresponds to the blocking call first encountered. If there are further callbacks with other blocking procedures in the two threads, the status bar may not correspond to the actual input pending, execution can appear to be taking place in one thread when it is really blocked in another, and the application can be confusing and misleading to the user. To avoid this confusion, you should try not to use blocking procedures in your callback routines. QuickWin will not accept more than one READ or INCHARQQ request through user callbacks from the same child window at one time. If one READ or INCHARQQ request is pending, subsequent READ or INCHARQQ requests will be ignored and -1 will be returned to the caller. If you have a child window that in some user scenario might call multiple callback routines containing READ or INCHARQQ requests, you need to check the return value to make sure the request has been successful, and if not, take appropriate action, for example, request again. This protective QuickWin behavior does not guard against multiple blocking calls through mouse selection of menu input options. As a general rule, using blocking procedures in callback routines is not advised, since the results can lead to program flow that is unexpected and difficult to interpret.
Callback Routines
All callback routines run in a separate thread from the main program. So, all multithread issues are in full force. In particular, sharing data, drawing to windows, and doing I/O must be properly coordinated and controlled. The Visual Fortran Sample POKER.F90 (in the ...\DF98\SAMPLES\QUICKWIN\POKER folder) is a good example of how to control access to shared resources. QuickWin callback routines, both for menu callbacks and mouse callbacks, should do a minimum of processing and then return. While processing a callback, the program will appear to be non-responsive because messages are not being serviced. This is why it is important to return quickly. If more processing time is needed in a callback, another thread should be started to perform this work; threads can be created by calling the Win32 API CreateThread. (For more information on creating and using threads, see
Using QuickWin
Page 40 of 40
Creating Multithread Applications.) If a callback routine does not start a new thread, the callback will not be reentered until it is done processing.
One thread does a READ or GETCHARQQ and is blocked until a character typed. The other thread (the main program) is in a loop doing useful work and checking in the loop to see if the other thread has received input.
For more information, see the Visual Fortran Samples PEEKAPP and PEEKAPP3 in the ...\DF98\SAMPLES\QUICKWIN folder.
Page 1 of 10
You can change the functions in a DLL without recompiling or relinking the applications that use them, as long as the functions' arguments and return types do not change. This allows you to upgrade your applications easily. For example, a display driver DLL can be modified to support a display that was not available when your application was created.
When general functions are placed in DLLs, the applications that share the DLLs can have very small executables. Multiple applications can access the same DLL. This reduces the overall amount of memory needed in the system, which results in fewer memory swaps to disk and improves performance.
Common blocks or module data placed in a DLL can be shared across multiple processes.
To build a DLL in the visual development environment, specify the Fortran Dynamic-Link Library project type. On the command line, specify the /dll option. You cannot make a QuickWin application into a DLL (see Using QuickWin) and QuickWin applications cannot be used with Fortran run-time routines in a DLL. This section describes the following aspects of creating Fortran DLLs:
l l l
Coding Requirements for Sharing Procedures in DLLs Coding Requirements for Sharing Data in DLLs Building and Using Dynamic-Link Libraries
Page 2 of 10
DLLs
A dynamic-link library (DLL) contains one or more subprograms that are compiled, linked and stored separately from the applications using them. Coding requirements include using the cDEC$ ATTRIBUTES DLLIMPORT and DLLEXPORT compiler directives. Variables and routines declared in the main program and in the DLL are not visible to each other unless you use DLLIMPORT and DLLEXPORT. This section discusses aspects of sharing subprogram procedures (functions and subroutines) in a Fortran DLL. To export and import each DLL subprogram: 1. Within your Fortran DLL, export each subprogram that will be used outside the DLL. Add a cDEC$ ATTRIBUTES DLLEXPORT directive to declare that a function, subroutine, or data is being exported outside the DLL. For example:
SUBROUTINE ARRAYTEST(arr) !DEC$ ATTRIBUTES DLLEXPORT :: ARRAYTEST REAL(4) arr(3, 7) INTEGER i, j DO i = 1, 3 DO j = 1, 7 arr (i, j) = 11.0 * i + j END DO END DO END SUBROUTINE
2. Within your Fortran application, import each DLL subprogram. Add a cDEC$ ATTRIBUTES DLLIMPORT directive to declare that a function, subroutine, or data is being imported from outside the current image. For example:
INTERFACE SUBROUTINE ARRAYTEST (rarray) !DEC$ ATTRIBUTES DLLIMPORT :: ARRAYTEST REAL rarray(3, 7) END SUBROUTINE ARRAYTEST END INTERFACE
The DLLEXPORT and DLLIMPORT options (for the cDEC$ ATTRIBUTES directive) define a DLL's interface. The DLLEXPORT property declares that functions or data are being exported to other images or DLLs, usually eliminating the need for a Linker module definition (.DEF) file to export symbols for the functions or subroutines declared with DLLEXPORT. When you declare a function, subroutine, or data with the DLLEXPORT property, it must be defined in the
Page 3 of 10
A program that uses symbols defined in another image (such as a DLL) must import them. The DLL user needs to link with the import LIB file from the other image and use the DLLIMPORT property inside the application that imports the symbol. The DLLIMPORT option is used in a declaration, not a definition, because you do not define the symbol you are importing. 3. Build the DLL and then build the main program, as described in Building and Using Dynamic-Link Libraries. Fortran and C applications can call Fortran and C DLLs provided the calling conventions are consistent (see Programming with Mixed Languages, especially Visual Fortran/Visual C++ Mixed-Language Programs). Visual Basic applications can also call Fortran functions and subroutines in the form of DLLs (see Programming with Mixed Languages, especially Calling Visual Fortran from Visual Basic). For more information:
l l
About building DLLs, see Building and Using Dynamic-Link Libraries. On sharing either common block or module data in a DLL, see Coding Requirements for Sharing Data in DLLs. About importing and exporting subprograms using a Sample program, see the Visual Fortran Sample TLS in folder ...\Df98\Samples\Advanced\Win32.
Declare the order, size, and data types of shared data consistently in the DLL and in all procedures importing the DLL exported data. If more than one thread or process can write to the common block simultaneously, use the appropriate features of the Windows operating system to control access to the shared data. Such features on Windows NT 4 and Windows 2000 systems include critical sections (for single process, multiple thread synchronization) and mutex objects (for multi-process
Page 4 of 10
Exporting and Importing Common Block Data Exporting and Importing Data Objects in Modules
Exporting and Importing Common Block Data Data and code in a dynamic-link library is loaded into the same address space as the data and code of the program that calls it. However, variables and routines declared in the program and in the DLL are not visible to one another unless you use the cDEC$ ATTRIBUTES DLLIMPORT and DLLEXPORT compiler directives. These directives enable the compiler and linker to map to the correct portions of the address space so that the data and routines can be shared, allowing use of common block data across multiple images. You can use DLLEXPORT to declare that a common block in a DLL is being exported to a program or another DLL. Similarly, you can use DLLIMPORT within a calling routine to tell the compiler that a common block is being imported from the DLL that defines it. To export and import common block data: 1. Create a common block in the subprogram that will be built into a Fortran DLL. Export that common block with a cDEC$ ATTRIBUTES DLLEXPORT directive, followed by the COMMON statement, associated data declarations, and any procedure declarations to be exported. For example:
!DEC$ ATTRIBUTES DLLEXPORT :: /X/ COMMON /X/ C, B, A REAL C, B, A END ...
If the Fortran DLL procedure contains only a common block declaration, you can use the BLOCK DATA statement:
BLOCK DATA T !DEC$ ATTRIBUTES DLLEXPORT :: /X/ COMMON /X/ C, B, A REAL C, B, A END
The Fortran procedure to be linked into a DLL can contain a procedure, such as the following:
SUBROUTINE SETA(I) !DEC$ ATTRIBUTES DLLEXPORT :: SETA, /X/ COMMON /X/ C, B, A REAL C, B, A
Page 5 of 10
2. Refer to the common block in the main image with a cDEC$ ATTRIBUTES DLLIMPORT directive, followed by the local data declarations and any procedure declarations defined in the exported DLL. For example:
PROGRAM COMMONX !DEC$ ATTRIBUTES DLLIMPORT:: SETA, COMMON /X/ C, B, A REAL C, B, A, Q EQUIVALENCE (A,Q) /X/
A = 0. I = 0 WRITE (6,*) 'In Main program before calling SETA...' WRITE (6,*) 'values of A and I:' , A, I CALL SETA(I) WRITE (6,*) 'In Main program after calling SETA...' WRITE (6,*) 'values of A and I:' , Q, I A = A + 1. I = I + 1 WRITE (6,*) 'In Main program after incrementing values' END PROGRAM COMMONX
3. Build the DLL and then build the main program, as described in Building and Using Dynamic-Link Libraries. Exporting and Importing Data Objects in Modules You can give data objects in a module the DLLEXPORT property, in which case the object is exported from a DLL. When a module is used in other program units, through the USE statement, any objects in the module with the DLLEXPORT property are treated in the program using the module as if they were declared with the DLLIMPORT property. So, a main program that uses a module contained in a DLL has the correct import attributes for all objects exported from the DLL. You can also give some objects in a module the DLLIMPORT property. Only procedure declarations in INTERFACE blocks and objects declared EXTERNAL or with cDEC$ ATTRIBUTES EXTERN can have the DLLIMPORT property. In this case, the objects are imported by any program unit using the module. If you use a module that is part of a DLL and you use an object from that module that does not have the DLLEXPORT or DLLIMPORT property, the results are undefined.
file://C:\TEMP\~hh5E82.htm
6/21/01
Page 6 of 10
On building a DLL, see Building and Using Dynamic-Link Libraries. On multithread programming, see Creating Multithread Applications.
The DLL requires an interface specification. The DLL is associated with a main project during execution, not during linking.
Building Dynamic-Link Libraries The DLL Build Output Checking the DLL Symbol Export Table Building Executables that Use DLLs DLL Sample Programs
Page 7 of 10
sets the /alignment option, which specifies whether padding is needed to ensure that exported data items are naturally aligned. For example, in the case of a common block containing four-byte variables, in the Project Setting dialog box you might specify: Open the appropriate workspace From the Project menu, click Settings Click the Fortran tab Select the Fortran Data category In the Common Element Alignment box, specify 4. 4. If you need to specify linker options, use the Linker tab of the Project Settings dialog box. 5. Build your Fortran DLL project.
The Microsoft visual development environment automatically selects the correct linker instructions for loading the proper run-time library routines (located in a DLL themselves). Your DLL is created as a multithreadenabled library. An import library (.LIB) is created for use when you link images that reference the DLL. To build the DLL from the command line: 1. If you build a DLL from the command line or use an exported makefile, you must specify the /dll option. For example, if the Fortran DLL source code is in the file f90arr.f90, use the following command line:
DF /dll f90arr.f90
A DLL named f90arr.dll. An import library, f90arr.lib, that you must link with applications that call your DLL.
If you also specify /exe:file or /link /out:file, you name a .DLL rather than an .EXE file (the default file extension becomes projectname.DLL instead of projectname.EXE) The /dll option selects as the default the DLL run-time libraries to support multithreaded operation. 2. If your DLL will export data, the procedures must be compiled and linked consistently. Consistently use the same /alignment option for the DLL export procedure and the application that references (imports) it. The goal is to specify padding to ensure that exported data items are naturally
file://C:\TEMP\~hh5E82.htm
6/21/01
Page 8 of 10
aligned, including common block data items and structure element alignment (structures in a module). 3. If you need to specify linker options, place them after the /link option on the DF command line. 4. Build the application. For example, if your DLL exports a common block containing four-byte variables, you might use the following command line (specify the /dll option):
DF /align:commons /dll dllfile.for
The /dll option automatically selects the correct linker instructions for loading the proper run-time library routines (located in a DLL themselves). Your DLL is created as a multithread-enabled library. For more information, see:
l l l l
The DLL Build Output Checking the DLL Symbol Export Table Building Executables that Use DLLs DLL Sample Programs
An import library (.LIB), which the linker uses to associate a main program with the DLL. The .DLL file containing the library's executable code.
Both files have the same basename as the library project by default. Your library routines are contained in the file projectname.DLL located in the default directory for your project, unless you specified another name and location. Your import library file is projectname.LIB, located in the default directory for your project.
Page 9 of 10
When you build the executable that imports the procedures or data defined in the DLL, you must link using the import library, check certain project settings or command-line options, copy the import library so the Linker can locate it, and then build the executable. To use the DLL from another image: 1. Add the import .LIB file with its path and library name to the other image. In the visual development environment, add the .LIB import library file to your project. In the Project menu, click Add to project, then Files. On the command line, specify the .LIB file on the command line. The import .LIB file contains information that your program needs to work with the DLL. 2. If your DLL exports data, consistently use the same project settings options in the Fortran Data category /alignment option as was used to create the DLL. In the Fortran Data compiler option category, specify the appropriate values for Common Element Alignment (common block data) and Structure Element Alignment (structures in a module). This sets the /alignment option, which specifies whether padding is needed to ensure that imported data items are naturally aligned. 3. In the Project Settings dialog box (Fortran tab), make sure the type of libraries specified is consistent with that specified for the Fortran DLL. 4. If you need to specify linker options: In the visual development environment, specify linker options in the Linker tab of the Project Settings dialog box. On the DF command line, place linker options after the /link option. 5. Copy the DLL into your path. For an application to access your DLL, it must be located in a directory on the search path or in the same directory as the main project. If you have more than one program accessing your DLL, you can keep it in a convenient directory identified in the environment path. If you have several DLLs, you can place them all in the same directory to avoid adding numerous directories to the path specification. When changing your path specification on a Windows Me, Windows 98, or Windows 95 system, you must restart the operating system for the change to take effect. On a Windows NT 4 or Windows 2000 system, you should log out and back in after modifying the system path. 6. Build the image that references the DLL.
Page 10 of 10
Make sure you have added the import library (created when you built the DLL file) to the project by (click the FileView tab). Like building other projects in the visual development environment, use the Build menu items to create the executable (see Defining Your Project).
Specify the import library at the end of the command line. If your DLL exports data that will be used by the application being built, specify the same /alignment options that were used to build the DLL. If you are building a main application, omit the /dll option. When building a Fortran DLL that references another DLL, specify the /dll option.
For example, to build the main application from the command line that references 4-byte items in a common block defined in dllfile.dll:
DF /align:commons mainapp.f90 dllfile.lib
The files associated with Sample DLLEXP2 are in the folder ...\DF98 \SAMPLES\DLL\DLLEXP2\. DLLEXP2 demonstrates how COMMON variables defined in a DLL can be shared between multiple programs. To build DLLEXP2, use the makefile. The Sample Loadexp1 shows how to dynamically load a DLL by using the LoadLibrary and GetProcAddress Win32 routines. The folder ...\DF98 \SAMPLES\DLL\Loadexp1\ contains the source files and a project workspace file.
Page 1 of 18
Deliver Fortran applications with a Windows Graphical User Interface (GUI). GUI applications typically use at least the Graphic Device Interface (GDI) and USER32 Win32 routines. Access all available Windows GDI calls with your Fortran applications. GDI functions use a 32-bit coordinate system, allowing coordinates in the +/-2 GB range, and performs skewing, reflection, rotation and shearing.
Only the Fortran Windows project type provides access to the full set of Win32 routines needed to create GUI applications. Windows projects are much more complex than other kinds of Visual Fortran projects. Before attempting to use the full capabilities of Windows programming, you should be comfortable with writing C applications and should familiarize yourself with the Win32 Software Development Kit (SDK). To build your application as a Fortran Windows application in the visual development environment, choose Fortran Windows Application from the list of Project types when you open a new project. When using the command line, specify the /winapp option to search the commonly used link libraries. Fortran Windows applications must use the DFWIN module or subset of DFWIN (see Calling Win32 Routines). The following Fortran Windows application topics are discussed:
l l l l l
Calling Win32 Routines Coding Requirements for Fortran Windows Applications Using Menus and Dialogs in SDI and MDI Fortran Windows Applications Sample Fortran Windows Applications Getting Help with Windows Programming
Page 2 of 18
This section describes general information about calling Win32 routines from Visual Fortran applications. It describes the following topics:
l l l l
l l l
A Comparison of Visual Fortran and Win32 Routines Including the Visual Fortran Interface Definitions for Win32 Routines Calling Win32 Routines Using the Visual Fortran Interface Definitions Special Naming Convention for Certain QuickWin and Win32 Graphics Routines Understanding Data Types Differences Examples Programs and Samples Additional Resources
Are intended to be called from the Fortran language. For example, character arguments are assumed to be Fortran character variables, not null-terminated C strings. May have QQ appended at the end of their names to differentiate them from equivalent Win32 operating system routines. Are described in the Visual Fortran Language Reference online documentation, in the A to Z Reference. The routine description lists the appropriate USE statement needed to include the interface definitions, such as USE DFLIB. Call appropriate Windows system (Win32) routines provided with the operating system. Are specific to Windows systems (one exception: most of the Visual Fortran Portability Library routines exist on most U*X systems).
In contrast, the Win32 Application Programming Interface (API) routines provided by the Windows operating system:
l
Are intended to be called from the C language. For example, character arguments are assumed to be null-terminated C strings. However, Visual Fortran provides interface block definitions that simplify calling Win32 routines from Visual Fortran (such as allowing you to specify Fortran data arguments as being passed by reference). To obtain these interface definitions, add the USE DFWIN line to your program (explained below).
Page 3 of 18
Often have multiple words appended together in their names, such as GetSystemTime. Are described in the Platform SDK online documentation. To look up a specific routine, use the Index or Search tabs in HTML Help viewer. The calling format is listed near the top of the page, followed by a description of the routine's arguments. The QuickInfo at the bottom of the Win32 routine documentation page lists the import library needed. Are also specific to Windows operating systems.
There are many groups of Win32 routines (see the Platform SDK and other resources). Win32 routines provide sophisticated window management, memory management, graphics support, threading, security, and networking. You can access many Win32 routines from any Fortran application, including Fortran Console and Fortran QuickWin applications. Only the Fortran Windows application project type provides access to the full set of Win32 routines needed to create GUI applications. Fortran Console applications are text-only applications. Fortran QuickWin applications allow you to build Windows style applications easily, but access only a small subset of the available Win32 API features. (Fortran QuickWin applications also allow you to use graphics.) For differences between Fortran QuickWin and Fortran Windows applications, see Comparing QuickWin with Windows-Based Applications.
To include the Win32 interface definitions, do one of the following: 1. Add the statement USE DFWIN to include all Win32 API routine definitions. The USE DFWIN statement makes all parameters and interfaces for most Windows routines available to your Visual Fortran program. Any program or subprogram that uses the Windows features can include the statement USE DFWIN, which is needed in each subprogram that makes graphics calls. Add the USE statement before any declaration statements (such as IMPLICIT NONE or INTEGER) or any other modules containing
Page 4 of 18
2. You can limit the type of parameters and interfaces for Windows applications to make compilation times faster. To do this, include only the subsets of the Win32 API needed in multiple USE statements (see the file ...\DF98\INCLUDE\DFWIN.F90). To locate the specific libraries for the routines being called, view the QuickInfo at the bottom of the routine page in the Platform SDK documentation, which lists the import library name. For example, to call only GetSystemTime, you need the interface definitions provided in kernel32.mod (binary form). To do this, add the following USE statement:
USE KERNEL32
If you want to further minimize compile time, add the ONLY keyword to the USE statement. For example:
USE KERNEL32, only: GetSystemTime, GetLastError
Whether the routine returns a result (a function) or not (subroutine). For example, the GetSystemTime routine calling format starts with VOID, so this routine should be called as a subroutine with a CALL statement.
2. If you are not sure about the data type of arguments or its function return value, you can examine the interface definitions in the appropriate .F90 file in ...\DF98\INCLUDE\. For example, to view the interface definition for GetSystemTime: 1. In a text editor (such as Notepad), open the file kernel32.f90 from ...\DF98\INCLUDE\.
Page 5 of 18
2. Search (or Find) the routine name (such as GetSystemTime) 3. View the interface definition and copy-and-paste parts of it into your source file text editor window. 3. If one of the arguments is a structure, you should look up the definition in DFWINTY.F90 in ...\DF98\INCLUDE\. For example, to view the data type definition for the T_SYSTEMTIME type used in GetSystemTime: 1. In a text editor (such as Notepad), open the file DFWINTY.F90 from ...\DF98\INCLUDE\. 2. Search (or Find) the data type name (such as T_SYSTEMTIME). 3. View the data type definition and copy-and-paste parts of it into your source file text editor window. Note the defined field names for later use. 4. Define a variable name to use the derived-type definition in your program, such as:
TYPE (T_SYSTEMTIME) MYTIME
5. Use the variable definition to call the Win32 routine. For example, the completed program follows:
! ! ! ! ! ! Getsystime.f90 file, shows how to call a Win32 routine Since the only routine called is GetSystemTime, only include interface definitions from kernel32.mod instead of all modules included by dfwin.f90. Type definitions are defined in DFWINTY, which is used within KERNEL32. PROGRAM Getsystime USE KERNEL32 TYPE (T_SYSTEMTIME) MYTIME CALL GetSystemTime(MYTIME) WRITE (*,*) 'Current UTC time hour and minute:', Mytime.wHour, Mytime.Wminute END PROGRAM
You might create a new Fortran Console (or QuickWin) application project, add the code shown above as a source file, build it, and view the result.
Special Naming Convention for Certain QuickWin and Win32 Graphics Routines
Most QuickWin routines have a QQ appended to their name ito differentiate them from equivalent Win32 operating system routines. However, a small group of QuickWin graphics routines have the same name as the Win32 routines,
Page 6 of 18
causing a potential naming conflict if your program unit includes both USE DFLIBS (which includes QuickWin routine interface definitions) and USE DFWIN (which includes Win32 API routine interface definitions). The QuickWin routines perform the same functions as the SDK routines but take a unit number, or use the unit in focus at the time of call, instead of taking a device context (DC) as one of their arguments. To handle this situation, a special MSFWIN$ prefix is used for the Win32 routines. These prefixed names must be used even if you only specify USE DFWIN. For example, Rectangle is a QuickWin routine, not a Win32 SDK routine, and you must use the name MSFWIN$Rectangle to refer to the SDK routine: QuickWin Routine ARC ELLIPSE FLOODFILL GETBKCOLOR GETPIXEL GETTEXTCOLOR LINETO PIE POLYGON RECTANGLE
1
Win32 API Routine MSFWIN$Arc MSFWIN$Ellipse MSFWIN$FloodFill MSFWIN$GetBkColor MSFWIN$GetPixel MSFWIN$GetTextColor MSFWIN$LineTo MSFWIN$Pie MSFWIN$Polygon MSFWIN$Rectangle MSFWIN$SelectPalette MSFWIN$SetBkColor MSFWIN$SetPixel MSFWIN$SetTextColor
Page 7 of 18
If an argument is described in the Platform SDK documentation as a pointer, then the corresponding Fortran interface definition of that argument would have the REFERENCE property (see the cDEC$ ATTRIBUTES directive). Older interface definitions use the Compaq Fortran pointer data type and pass the address of the argument. Pointer arguments are currently 32-bit (4 bytes) in length. A future version of the Windows operating system will support 64-bit (8-byte) pointer arguments. Be aware that Fortran character variables need to be null-terminated. This code shows the extension of using the null-terminator for the string in the Fortran DATA statement (see C Strings):
DATA forstring /'This is a null-terminated string.'C/
Page 8 of 18
The structures in WINDOWS.H have been converted to derived types in DFWINTY. Unions in structures are converted to union/maps within the derived type. Names of components are unchanged. Bit fields are converted to Fortran's INTEGER(4). Functions accessing bit fields are contained in the DFWIN.F90 module with names of the form:
StructureName$BitfieldName
These bit field functions take an integer argument and return an integer. All bit fields are unsigned integers. The following shows the WINDOWS.H definition. It is followed by an example program where Win32 structures are represented as Fortran derived types.
WINDOWS.H Definition typedef struct _LDT_ENTRY { WORD LimitLow; WORD BaseLow; union { struct { BYTE BaseMid; BYTE Flags1; BYTE Flags2; BYTE BaseHi; } Bytes; struct { DWORD DWORD DWORD DWORD DWORD DWORD DWORD DWORD DWORD DWORD } Bits; } HighWord; } LDT_ENTRY, BaseMid : 8; Type : 5; Opl : 2; Pres : 1; LimitHi : 4; Sys : 1; Reserved_0 : 1; Default_Big : 1; Granularity : 1; BaseHi : 8; *PLDT_ENTRY;
Note that _LDT_ENTRY and PLDT_ENTRY do not exist in the Fortran definition. Also note that Bits.xxx is not the same as the C version. In the Fortran case, the bit field functions must be used. For example, the C variable yyy.HighWord.Bits.BaseHi is represented in Fortran by LDT_ENTRY$BaseHi (ldtentry%HighWord%bits). The following Fortran example shows the use of the corresponding Fortran data definitions and the use of a bit extraction utility routine:
Program Test type T_LDT_ENTRY$HIGHWORD_BYTES sequence BYTE BaseMid BYTE Flags1
Page 9 of 18
Bytes
HighWord
In addition to using the TYPE statement, you can use the STRUCTURE statement as shown by the following example. If you have embedded data types, the data type definitions are easier to read. Also, selecting the components is more natural. For example:
program struct STRUCTURE /T_LDT_ENTRY/ integer(2) LimitLow integer(2) BaseLow UNION MAP STRUCTURE /f_Bytes/ Bytes BYTE BaseMid BYTE Flags1 BYTE Flags2 BYTE BaseHi END STRUCTURE END MAP MAP STRUCTURE /f_Bits/ Bits
Page 10 of 18
You can create a data object defined by a STRUCTURE statement in two ways, by using one of the following:
l l
The example program shows how to access a field in a structured data object, for example:
I4=ldtentry%Bits%Bits
Fortran Console applications in the folder ...\Df98\Misc, such as Console and Forprint. Fortran QuickWin applications in the folder ...\Df98\QuickWin, such as Conapp, Cleanwin, and Testscrl. Fortran Windows applications the folder ...\Df98\Advanced\Win32, such as Generic and Bounce.
Additional Resources
For more information about Win32 API routines and Windows programming, see the following:
l l
Platform SDK online HTML Help title Microsoft Win32 Developer's Reference Library by David Iseminger (Microsoft Press) Programming Windows by Charles Petzold (Microsoft Press)
Page 11 of 18
Windows NT Win32 API SuperBible by Richard Simon (Waite Group Press) Win32 Programming by Brent Rector and Joseph Newcomer (AddisonWesley) Win32 System Programming by Johnson Hart (Addison-Wesley)
Visual Fortran provides an online bookstore that lists suggested titles: 1. 2. 3. 4. 5. Open the Visual Fortran home page (http://www.compaq.com/fortran) Click Bookstore in the left margin Click the link to the Compaq Fortran Online Bookstore Click the link for the appropriate area of interest, such as Win32 View the descriptions of the appropriate books
General Coding Requirements: WinMain Function and USE Statements Code Generation Options Using the Fortran Windows Application Wizard Single Document Interface (SDI) or Multiple Document Interface (MDI) Applications Dialog-Based Applications
In a program that includes a WinMain function, no program unit can be identified as the main program with the PROGRAM statement.
Page 12 of 18
2. The statement USE DFWIN or other appropriate USE statements The USE DFWIN statement makes all parameters and interfaces for nearly all Windows public routines available to your Visual Fortran program. Any program or subprogram that uses the Windows features must include the statement USE DFWIN, which must appear in each subprogram that makes graphics calls, before any declaration statements (such as IMPLICIT NONE or INTEGER) or any other modules containing declaration statements. If you want to limit the type of parameters and interfaces for Windows applications or if unresolved references occur when linking your Fortran Windows application, see Calling Win32 Routines. 3. Data declarations for the WinMain function arguments. 4. Application-dependent code (other USE statements, variable declarations, and then executable code). For example, the first lines of the Visual Fortran Sample named Generic uses the following free-form source code and conditional ia32 and ia64 code:
integer function WinMain( hInstance, hPrevInstance, lpszCmdLine, nCmdShow ) !DEC$ IF DEFINED(_M_IX86) !DEC$ ATTRIBUTES STDCALL, ALIAS : '_WinMain@16' :: WinMain !DEC$ ELSE !DEC$ ATTRIBUTES STDCALL, ALIAS : 'WinMain' :: WinMain !DEC$ ENDIF use dfwin integer integer integer integer . . . hInstance hPrevInstance nCmdShow lpszCmdLine
This Sample uses the IF Directive Construct and a predefined preprocessor symbol _M_IX86 to generate portable conditional code. For a description of predefined preprocessor symbols (such as _M_IX86 and _M_IA64), see the /define compiler option. DFWIN.F90 includes a Fortran version (a subset) of the Win32 WINDOWS.H header file (see Calling Win32 Routines).
Page 13 of 18
the Fortran Windows Application Wizard (AppWizard) appears, allowing you to select whether the new project being created will have source code, type of tutorial source code, and specify other options:
If selected, you can specify whether the source file will contain template-like source statements typically used by one of the Fortran Windows applications subtypes:
l
Single document interface (SDI) or multiple document interface (MDI) applications Dialog-based applications
If your application (SDI, MDI, or dialog-based) will use ActiveX controls in a dialog box, if you check the box labeled "This project will use ActiveX controls," the Fortran Windows AppWizard will add additional template code for supporting ActiveX controls in your dialog boxes. To select project options that request that the project be linked against dynamic-link libraries (instead of static libraries), click the check box labeled "This project will be linking against one or more Fortran DLL Import libraries."
Page 14 of 18
routines and requires the statement USE DFWIN and usually USE DFLIB. SDI applications display a single window, whereas MDI application can display multiple windows (a main frame window with one or more child windows that appear within the frame window). For example, select the MDI option from the Fortran AppWizard screen. After you build and run the application (without changing the source files), the following screen might appear after you create two child window by clicking New from the File menu twice:
If you selected the SDI option from the Fortran AppWizard screen and built and ran the application, you could not create child windows within the main window. For more information:
l
On using menus and dialogs from SDI and MDI Fortran Windows applications, see Using Menus and Dialogs in SDI and MDI Fortran Windows Applications About SDI and MDI Samples that use the Fortran Windows project type, see Sample Fortran Windows Applications.
Dialog-Based Applications
Dialog applications use a dialog box for the application's main window. Creating these applications requires some knowledge of the Win32 routines API, but considerably less than for a SDI or MDI application. These applications call
Page 15 of 18
certain Visual Fortran library routines and requires the statements USE DFWIN and USE DFLOGM. Dialog-based applications usually do not have menus. For example, select the Dialog-based applications from the Fortran AppWizard screen. After you build and run the application (without changing the source files), the following dialog box appears:
On using dialogs, see Using Dialogs. About dialog Samples that use the Fortran Windows project type, see Sample Fortran Windows Applications.
Using Menus and Dialogs in SDI and MDI Fortran Windows Applications
This section describes the following topics:
l l l l
Creating the Menu Using the Menu Handling Menu Messages Using Dialogs in an SDI or MDI Application
Creating the Menu When you create a new SDI or MDI application using the Fortran Windows AppWizard, a default menu bar is created for you. The default menu bar contains many of the menu entries that are common to Windows applications. You can modify the default menu, or create a new menu, by using the Menu Editor, which is one of the Developer Studio Resource Editors. To create a new menu resource (menu bar): 1. Click Resource from the Insert menu 2. Select Menu as the resource type
Page 16 of 18
The menu bar consists of multiple menu names, where each menu name contains one or more items. You can use the Menu Editor to create submenus (select the pop-up property). To edit an existing menu: 1. Select the ResourceView 2. Expand the Menu item in the list of resource types 3. Double click on the menu mame For more information about the menu resource editor, see the Visual C++ User's Guide section on Resource Editors. Using the Menu To use a menu bar in your Fortran application, you must load the menu resource and use it when creating the main window of the application. Code to do this is created automatically by the Fortran Windows AppWizard. The code that loads the menu resource is:
ghMenu = LoadMenu(hInstance, LOC(lpszMenuName))
Handling Menu Messages Windows sends a WM_COMMAND message to the main window when the user selects an item from the menu. The wParam parameter to the WM_COMMAND message contains:
l l
The low-order word specifies the identifier of the menu item The high-order word specifies either 0 if the message is from a menu item, or 1 if the message is the result of an accelerator key. It is usually not important to distinguish between these two cases, but you must be careful to compare against only the low-order word as in the example below.
Page 17 of 18
For example, the following code from the main window procedure generated by the Fortran Windows AppWizard handles the WM_COMMAND messages from the File menu Exit item and the Help menu About item:
! WM_COMMAND: user command case (WM_COMMAND) select case ( IAND(wParam, 16#ffff ) ) case (IDM_EXIT) ret = SendMessage( hWnd, WM_CLOSE, 0, 0 ) MainWndProc = 0 return case (IDM_ABOUT) lpszName = "AboutDlg"C ret = DialogBoxParam(ghInstance,LOC(lpszName),hWnd,& LOC(AboutDlgProc), 0) MainWndProc = 0 return ...
For advanced techniques with using menus, refer to the online Platform SDK section on User Interface Services, for subsections: Windows User Interface, Resources, Menus. Using Dialogs in an SDI or MDI Application An Fortran Windows SDI or MDI application that uses dialogs has the choice of using:
l l
The Visual Fortran Dialog routines The native Win32 APIs for creating dialog boxes
For any particular dialog box, you should use either the Visual Fortran Dialog routines or the native Win32 dialog box APIs. For example, if you create a dialog box using Win32 APIs, you cannot use the Visual Fortran dialog routines to work with that dialog box. You should Note, for example, that the code generated by the Fortran Windows AppWizard uses the native Win32 APIs to display the About dialog box. For more information:
l l
On using the Visual Fortran Dialog routines, see Using Dialogs On using the Win32 APIs, see the online Platform SDK section on User Interface Services, for subsections: Windows User Interface, Windowing, Dialog Boxes.
Page 18 of 18
The Visual Fortran Samples ...\DF98\SAMPLES\ADVANCED folder contains many Fortran Windows applications that demonstrate Windows functionality or a particular Win32 function. Each sample application is in separate folder. Users unfamiliar with full Windows applications should start by looking at:
l
Sample SDI and MDI Fortran Windows Samples in ...\DF98 \SAMPLES\ADVANCED\WIN32, such as Generic, Platform, or Polydraw. Sample dialog Fortran Windows Samples in ...\DF98\SAMPLES\DIALOG, such as THERM or MMPLAYER. For more information about coding reqirements for dialog boxes and using the Dialog Resource editor, see Using Dialogs.
Page 1 of 20
Portability considerations Choosing Your Development Environment with Visual Fortran Selecting a Program Type that you can build Structuring Your Program Special Design Considerations Using the Special Features of Microsoft Windows with your programs
Portability
This section presents topics to help you understand how language standards, operating system differences, and computing hardware influence your use of Visual Fortran and the portability of your programs. Your program is portable if you can implement it on one hardware-software platform and then move it to additional systems with a minimum of changes to the source code. Correct results on the first system should be correct on the additional systems. The number of changes you might have to make when moving your program varies significantly. You might have no changes at all (strictly portable), or so many (non-portable customization) that it is more efficient to design or implement a new program. Most programs in their lifetime will need to be ported from one system to another, and this section can help you write code that makes this easy. For information on special library routines to help port your program from one system to another, see Portability Library. For more information:
l l l
Page 2 of 20
standards. ANSI (American National Standards Institute) and ISO (International Standards Organization) are the primary organizations that develop and publish the standards. The major Fortran language standards are:
l
FORTRAN IV American National Standard Programming Language FORTRAN, ANSI X3.91966. This was the first attempt to standardize the languages called FORTRAN by many vendors.
FORTRAN 77 American National Standard Programming Language FORTRAN, ANSI X3.91978. This standard added new features based on vendor extensions to FORTRAN IV and addressed problems associated with large-scale projects, such as improved control structures.
Fortran 90 American National Standard Programming Language Fortran, ANSI X3.1981992 and International Standards Organization, ISO/IEC 1539: 1991, Information technology -- Programming languages -- Fortran. This standard emphasizes modernization of the language by introducing new developments. For information about differences between Fortran 90 and FORTRAN 77, see Features of Fortran 90 or the printed Compaq Fortran Language Reference Manual.
Fortran 95 American National Standard Programming Language Fortran and International Standards Organization, ISO/IEC 1539-1: 1997(E), Information technology -- Programming languages -- Fortran. This recent standard introduces certain language elements and corrections into Fortran 90. Fortran 95 includes Fortran 90 and most features of FORTRAN 77. For information about differences between Fortran 95 and Fortran 90, see Features of Fortran 95 or the printed Compaq Fortran Language Reference Manual.
Although a language standard seeks to define the form and the interpretation uniquely, a standard might not cover all areas of interpretation. It might also include some ambiguities. You need to carefully craft your program in these cases so that you get the answers that you want when producing a portable program. For more information:
Page 3 of 20
l l
The Fortran 90 standard added many new intrinsic procedures to the language. Programs that conformed to the FORTRAN 77 standard may include nonintrinsic functions or subroutines having the same name as new Fortran 90 procedures. Some processors include nonstandard intrinsic procedures that might conflict with procedure names in your program.
If you do not explicitly declare the external procedures and the name duplicates an intrinsic procedure, the processor calls the intrinsic procedure, not your external routine. For more information on how the Fortran compiler resolves name definitions, see Resolving Procedure References.
Compiler Optimizations
Many Fortran compilers perform code-generation optimizations to increase the
Page 4 of 20
speed of execution or to decrease the required amount of memory for the generated code. Although the behaviors of both the optimized and nonoptimized programs fall within the language standard specification, different behaviors can occur in areas not covered by the language standard. Compiler optimization especially can influence floating-point numeric results. The Visual Fortran compiler can perform optimizations to increase execution speed and to improve floating-point numerical consistency. For a summary of optimization levels, see Optimization Levels and Other Options Related to Optimization. Floating-point consistency refers to obtaining results consistent with the IEEE binary floating-point standards (see the /fltconsistency option). Unless you properly design your code, you might encounter numerical difficulties when you optimize for fastest execution. The /nofltconsistency option uses the floating-point registers, which have a higher precision than stored variables, whenever possible. This tends to produce results that are inconsistent with the precision of stored variables. The /fltconsistency option (also set by /Oxp) can improve the consistency of generated code by rounding results of statement evaluations to the precision of the standard data types, but it does produce slower execution times.
Operating System
The operating system envelops your program and influences it both externally and internally. To achieve portability, you need to minimize the amount of operating-system-specific information required by your program. The Fortran language standards do not specify this information. Operating-system-specific information consists of nonintrinsic extensions to the language, compiler and linker options, and possibly the graphical user interface of Windows. Input and output operations use devices that may be systemspecific, and may involve a file system with system-specific record and file structures. The operating system also governs resource management and error handling. You can depend on default resource management and error handling mechanisms or provide mechanisms of your own. For information on special library routines to help port your program from one system to another, see Portability Library. The minimal interaction with the operating system is for input/output operations and usually consists of knowing the standard units preconnected for input and output. You can use default file units with the asterisk (*) unit specifier. To increase the portability of your programs across operating systems, consider
Page 5 of 20
Do not assume the use of a particular type of file system. Do not embed filenames or paths in the body of your program. Define them as constants at the beginning of the program or read them from input data. Do not assume a particular type of standard I/O device or the "size" of that device (number of rows and columns). Do not assume display attributes for the standard I/O device. Some environments do not support attributes such as color, underlined text, blinking text, highlighted text, inverse text, protected text, or dim text.
Size of Basic Types Bit, Byte, and Word Characteristics Transportability of Data
Page 6 of 20
Number of Bytes 1
2 Depending on default integer size (set by the /integer_size compiler option or equivalent directive) , INTEGER and LOGICAL can have 2, 4, or 8 bytes; default allocation is 4 bytes. Depending on default real size (set by the /real_size compiler option or equivalent directive), REAL can have 4 or 8 bytes; default allocation is 4 bytes.
INTEGER(4), REAL (4), LOGICAL(4) INTEGER(8), LOGICAL(8) COMPLEX DOUBLE PRECISION, REAL (8), COMPLEX(4) DOUBLE COMPLEX, COMPLEX(8) CHARACTER(n) Structures RECORD
4 8 Depending on default real, COMPLEX can have 8 or 16 bytes; default allocation is 8 bytes. 8
16 n Size of derived type (can be affected by the PACK directive) Size of record structure (can be affected by the PACK directive)
Page 7 of 20
consecutive subcomponents (bytes) of a word or are consecutive, multiple words. However, when transporting binary data among disparate systems -either by intermediate storage medium (disk, tape) or by direct connection (serial port, network) -- problems arise from different definitions of serial bit and serial byte order. For simplicity, the following discussion considers only byte order within a word, since that is the usual case of difficulty. (For more information, refer to "On Holy Wars and a Plea for Peace" by Danny Cohen, IEEE Computer, vol. 14, pp. 48-54, 1981.) For more information:
l l l
Big End or Little End Ordering Binary Representations Declaring Data Types
Data types stored as subcomponents (bytes stored in words) end up in different locations within corresponding words of the two conventions. The following figure illustrates the difference between the representation of several data types in the two conventions. Letters represent 8-bit character data, while numbers represent the 8-bit partial contribution to 32-bit integer data. Character and Integer Data in Words: (a) Big End, (b) Little End
Page 8 of 20
If you serially transfer bytes now from the Big End words to the Little End words (BE byte 0 to LE byte 0, BE byte 1 to LE byte 1, ...), the left half of the figure shows how the data ends up in the Little End words. Note that data of size one byte (characters in this case) is ordered correctly, but that integer data no longer correctly represents the original binary values. The right half of the figure shows that you need to swap bytes around the middle of the word to reconstitute the correct 32-bit integer values. After swapping bytes, the two preceding figures are identical. Data Sent from Big to Little: (a) After Transfer, (b) After Byte Swaps
You can generalize the previous example to include floating-point data types and to include multiple-word data types. The following table summarizes the ordering nature of several common processors. Ordering Nature of Processors
Page 9 of 20
Processor Intel 80486, Pentium Series Compaq Alpha and VAX Motorola 680XX IBM Mainframes
The important result is that portable, serial transport of 8-bit character data between most systems is possible with little or no knowledge about the ordering nature of each system. For more information on big and little endian data and Visual Fortran unformatted data conversion capabilities, see Converting Unformatted Numeric Data.
Binary Representations
The discussion in Big End or Little End Ordering stresses 8-bit character data because you might encounter hardware that uses a different representation of binary data. The Visual Fortran system uses the two's-complement representation of negative binary integers. You might encounter a system that uses a signed magnitude representation, a one's complement representation, or a biased (excess) representation. Additionally, the bit representation of binary floating-point numbers is not unique. If you transport binary data to or from a different system, you need to know the respective representations to convert the binary data appropriately.
Transportability of Data
You can achieve the highest transportability of your data by formatting it as 8bit character data. Use a standard character set such as the ASCII standard for
Page 10 of 20
encoding your character data. Although this practice is less efficient than using binary data, it will save you from shuffling and converting your data. If you are transporting your data by means of a record-structured medium, it is best to use the Fortran sequential formatted (as character data) form. You can also use the direct formatted form, but you need to know the record length of your data. Remember also that some systems use a carriage return-linefeed pair as an end-of-record indicator, while other systems use linefeed only. If you use either the direct unformatted or the sequential unformatted form, there might be system-dependent values embedded within your data that complicate its transport. Implementing a strictly portable solution requires a careful effort. Maximizing portability may also mean making compromises to the efficiency and functionality of your solution. If portability is not your highest priority, you can use some of the techniques that appear in later sections to ease your task of customizing a solution. For more information on big and little endian data and unformatted data conversion, see Converting Unformatted Numeric Data.
Page 11 of 20
Because software development is an iterative process, it is important to be able to move quickly and efficiently to various locations in your source code. If you use the visual development environment to compile and link your programs, you can call up both the description of the error message and the relevant source code directly from the error messages in the output window. You also use the visual development environment text editor to view and control execution of your program with the integrated source level debugger. Finally, when you use the project browser to locate routines, data elements, and references to them, the visual development environment uses its editor to go directly to the source code. When you build programs from the console, you are in complete control of the build tools. If you choose to, you can customize how your program is built by your selection of compiler and linker options. Compiler and linker options are described in Compiler and Linker Options. Even if you choose to edit and build your program from the command line, you can still use the visual development environment debugger and browser after your program has compiled and linked cleanly (see Preparing Your Program for Debugging). Finally, you can run the profiler to produce a text report of your program's execution statistics either from the command-line console or from the visual development environment.
Console applications Standard graphics applications QuickWin graphics applications Windows applications
You can also create a Fortran COM server project or a library project that contains subprograms (functions or subroutines) called from your main application:
l l
Code that works in one application may not work in others. For example, graphics calls are not appropriate in a Fortran console application. Fortran console applications are the most portable to other systems because
Portability and Design Considerations they are text-only and do not support graphics.
Page 12 of 20
With Fortran standard graphics (QuickWin single document) applications, you can add graphics to your text without the additional overhead of menus and other interface features of typical programs for Windows. Fortran QuickWin (QuickWin multiple document) graphics applications provide a simple way to use some features of Windows in a Visual Fortran program with graphics. Fortran Windows applications give users full access to the Win32 Application Programming Interface (API), giving you a larger set of functions than QuickWin offers. With Windows applications, you can access low-level system services directly, or access higher level system services such as OpenGL. None of the graphics functions in Visual Fortran, except for those in the OpenGL library, are directly portable to operating systems offered by other vendors. A graphical interface does, however, offer certain advantages to the application designer and to the person who will use the program. The choice of what kind of program to build is a trade-off between performance, portability, ease of coding, and ease of use. The advantages and disadvantages of each type of application are summarized in the following sections. All four kinds of main applications can be maximized, minimized, resized, and moved around the screen when displayed in a window. If the drawing area of a window in your application is larger than the window in which it is displayed, scroll bars are automatically added to the bottom and right edges of the window. You can write any of the applications with one section of the program beginning execution before another has completed. These threads of execution run either concurrently on a computer with one processor or simultaneously on a computer with multiple processors. (See Creating Multithread Applications.) For more information on the Visual Fortran application project types, see Types of Projects.
Page 13 of 20
Advantages of Modules
One way to reduce potential confusion when you use the same source code in several projects is to organize the routines into modules. There are two main uses for modules in Visual Fortran:
l
Internal encapsulation A single complex program can be made up of many modules. Each module can be a self-contained entity, incorporating all the procedures and data required for one of your program's tasks. When a task is encapsulated, it is easy to share the code between two different projects. In this case, all the modules should be included in the main project directory. If many projects all share the same module, the module should reside in only one directory. All projects that use it should specify the /I compiler option to indicate the location of the module.
External modules If you use a module provided from an outside source, you need only the .MOD file at compile time, and the .OBJ file at link time. Use the /[no]include[path] (or /Ipath) command line option (or the INCLUDE environment variable) to specify the location of these files, which will probably not be the same as your project directory.
During the building of a project, the compiler scans the project files for dependencies. If you specify the /[no]include[path] (or /Ipath) command line option or the INCLUDE environment variable, the compiler is able to find the external modules.
Page 14 of 20
Store precompiled module files, with the extension .MOD, in a directory included in the path. When the compiler sees the USE statement in a program, it finds the module based on the name given in the USE statement, so there is no need to maintain several copies of the same source or object code. Modules are excellent ways to organize programs. You can set up separate modules for:
l l l
Commonly used routines Data definitions specific to certain operating systems System-dependent language extensions
From the visual development environment, build a Fortran Static Library project type. From the command line, use the LIB command.
After you have created a static library, you can use it to build any of the other types of Visual Fortran projects. For more information:
l
Page 15 of 20
Porting Fortran Source Between Systems Mixed-Language Issues Porting Data Between Systems
Page 16 of 20
l l
l l
Isolate system-dependent code into separate modules. Maintain distinct modules with similar functionality for each separate platform. In your main program, use only language extensions that will compile on both platforms, putting system-dependent code into modules. Place language extension subsets into modules. If you use Microsoft compiler directives, replace the older $directive format with the !DEC$ directive format, because this will be ignored by other systems. Specify data precision, for integers and logicals as well as for floating-point numbers when the size matters. If you do not explicitly specify KIND for variables, this could be the source of problems if one system uses a default of (KIND=2) for integers, while your program assumes (KIND=4). Conversely, if the size of a variable is not significant, avoid specifying data precision. Code that does specify precision will run slower on systems that do not use the same default integer and real sizes. Avoid using algorithms that exhibit floating-point instability. For information on handling floating-point numbers, see The Floating-Point Environment. Specify equivalent floating-point precision on each platform. Specify the appropriate attributes when defining routines and data that will be interacting with code written in Microsoft Visual C/C++ or assembly language.
On porting code between systems, see Portability. On compatibility with Compaq Fortran on Alpha platforms, see Compatibility with Compaq Fortran on Other Platforms
Choosing a Language Extension Subset The Visual Fortran compiler supports extensions used on a variety of platforms, plus some that are specific to Visual Fortran. Because there are Fortran compilers for many different computers, you might need to move your source code from one to another. If the trip is one-way and it is permanent, you can simply change the code to work on the second platform. But if you need to make sure you can move the code where ever it is needed, you must be aware of the extensions to Fortran that are supported on each platform. You can use some of the Visual Fortran compiler options to help you write portable code. For example, by specifying ANSI/ISO syntax adherence (/stand option) in the Project Settings (Fortran tab) dialog box or on the command line, you can have the compiler enforce Fortran 90 or 95 syntax. Code that compiles cleanly with this option set is very likely to compile cleanly on any other computer with a Fortran compiler that obeys strict Fortran syntax. If you choose to use platform-specific extensions, you need to note whether
Page 17 of 20
there are any differences in how those extensions are implemented on each computer, and use only those features that are identical on both. (For more information, see Portability.) The default is to compile with the full set of extensions available. Because Visual Fortran compiler directives look like standard Fortran comments (such as !DEC$ directive), programs that use directives can compile on other systems. They will, however, lose their function as compiler directives. Floating-Point Issues Floating-point answers can differ from system to system, because different systems have different precisions and treat rounding errors in different ways. One programming practice that can be a serious source of floating-point instability is performing an IF test (either obvious or implied) that takes some action if and only if a floating-point number exactly equals a particular value. If your program contains code like this, rewrite the code to a version that is stable in the presence of rounding error. For more details, see The Floating-Point Environment. Another source of floating-point instability is the use of mathematical algorithms that tend to diminish precision. Incorrect answers can result when the code is moved to a system with less precision. For more information, see The FloatingPoint Environment. One way of making all REAL variables on one system DOUBLE PRECISION on another is to use modules to declare explicit data types for each system. Specify a different KIND parameter in each module. Another way is to add an include file that declares explicit data types on each system in all source files.
Mixed-Language Issues
You can combine object modules generated by Visual Fortran with object files from compilers for 32-bit Windows that compile other languages (such as Microsoft Visual C++, or Microsoft MASM), so long as the compilers use the COFF object module format used by Microsoft. You need to respect certain calling, naming, and argument-passing conventions when combining object modules from different languages. These conventions are discussed in Programming with Mixed Languages.
Page 18 of 20
If you try to transfer unformatted binary data between systems, you need to be aware of the different orders (low-order byte first or high-order byte first) in which different systems store bytes within words. If you need to transfer unformatted binary data, review:
l l l
Portability Compatibility with Compaq Fortran on Other Platforms Converting Unformatted Numeric Data
Built-in Benefits of Windows Single or Multithread Program Execution Dialog Boxes QuickWin and Windows Programs
Page 19 of 20
hanging the computer. If your program hangs, you can easily and safely stop it by using the Windows task manager. (For information about using this or any other feature of Windows, see the manuals that came with the operating system.) Because you can use one application while another continues to execute, you can make better use of your own time. For example, you can use the visual development environment to edit the source for one project while another project is building, or use Microsoft Excel to prepare a graph for data that your program is busy producing. And if your computer has multiple processors and you are using Windows NT 4 or Windows 2000, the computation-intensive program producing your data might be executing on an otherwise idle processor, making it less likely that your other work will slow it down.
Dialog Boxes
Visual Fortran gives you an easy way to create simple dialog boxes that can be used for data entry and application control. Dialogs are a user-friendly way to get and process input. As your application executes, you can make a dialog box appear on the screen and the user can click on a button or scroll bar to enter data or choose what happens next. You can add dialog boxes to any Fortran application, including Fortran Windows, Fortran QuickWin, and Fortran Console applications. You design your dialog with the Resource Editor, and drive them with a combination of the dialog routines, such as DLGSET, and your own subroutines. A complete discussion of how to design and use dialog boxes is given in Using Dialogs.
Page 20 of 20
Using Dialogs
Page 1 of 53
Using Dialogs
Dialogs are a user-friendly way to solicit application control. As your application executes, you can make a dialog box appear on the screen and the user can click on a dialog box control to enter data or choose what happens next. With the dialog routines provided with Visual Fortran, you can add dialog boxes to your application. These routines define dialog boxes and their controls (scroll bars, buttons, and so on), and call your subroutines to respond to user selections. There are two types of dialog boxes:
l
Modal dialog boxes, which you can use with any Fortran project type, including Fortran Windows, Fortran QuickWin (multiple doc.), Fortran Standard Graphics (QuickWin single doc.), Fortran Console, Fortran DLL, and Fortran Static library project types. Modeless dialog boxes, which are typically used with the Fortran Windows project type.
When your program displays a modal dialog box (any project type), the user must explicitly enter data and close the dialog box before your application resumes execution. When your program displays a modeless dialog box, your application continues executing. Unlike a modal dialog box, the user can switch between the modeless dialog box and the other windows in the application. There are two steps to make a dialog: 1. Specify the appearance of the dialog box and the names and properties of the controls it contains. 2. Write an application that activates those controls by recognizing and responding to user selections. This section covers the following topics:
l l l l l l
Using the Resource Editor to Design a Dialog Writing a Dialog Application Dialog Routines Dialog Controls Using Dialog Controls Using ActiveX Controls
Using Dialogs
Page 2 of 53
You design the appearance of the dialog box, choose and name the dialog controls within it, and set other control properties with the Dialog Editor. The Dialog Editor is one of the Resource Editors provided by the visual development environment (Developer Studio). Developer Studio contains other resource editors for editing icons, bitmaps, menus, and so on. For information on the other resource editors, see "Resource Editors" section in the Visual C++ User's Guide . A program's resources are defined in a resource file (typically with a .rc extension). A Developer Studio project typically contains a single resource file. The contents of the resource file are displayed in the ResourceView. The resource file can be created by one of the following:
l
When you create a project and use one of the Fortran AppWizards (for example, when using the Fortran Windows Application AppWizard). When you save the resources that you define using one of the Resource Editors.
If you create the resource file from the resource editors, be sure to add the resource file to your project. After you do this, the resources will be displayed in the ResourceView. It is possible to include additional resource files in a project (see Including Resources Using Multiple Resource Files). This section describes the steps needed to design a dialog box, and uses as an example a dialog box that converts temperatures between Celsius and Fahrenheit. The code in the example is explained as you read through this chapter. In this section, we include the dialog box in a Fortran Console project. You can use the completed Visual Fortran sample in the ...\Samples\Dialog\Temp directory or you can create your own application. To create a Fortran Console application:
l l l l
From the File menu, click New... Click the Project tab From the list of project types, select Fortran Console Application Enter TEMP as the name for the project, verify where the project will be created, and click OK In the Fortran Console Application - Step 1 of 1 dialog box, choose A simple project and click Finish Click OK after reading the displayed summary information. The project workspace is opened, displaying the FileView pane. In the FileView pane, click the plus sign (+) next to the project name to display the categories of source files.
Using Dialogs
Page 3 of 53
To open the dialog resource editor: 1. From the Insert menu, choose Resource. 2. From the list of possible resources, choose Dialog. 3. Click the New button. The dialog editor (one of the resource editors) appears on the screen as shown below. Dialog Editor Sample 1
A blank dialog box appears at the left and a toolbar of available controls appears on the right. If the Controls toolbar does not appear: 1. Click Customize in the Tools menu 2. Click the Toolbars tab 3. Select (check) Controls If you place the cursor over a control on the toolbar, the name of the control appears. The Controls toolbar items that are supported by Visual Fortran follow: Check box Group box List box Progress bar
Using Dialogs Radio button Scroll bar, vertical Spin control Tab control Scroll bar, horizontal Slider Static text
Page 4 of 53
You can also add ActiveX controls to your dialog box (see Using ActiveX Controls). To add controls to the dialog box: 1. Point at one of the available controls on the Control toolbar, hold down the left mouse button and drag the control to the dialog box. 2. Place the dialog control where you want it to be on the dialog box and release the mouse button. You can delete controls by selecting them with the mouse, then pressing the Delete (or DEL) key. The following figure shows the dialog box after adding two Static text lines (currently say Static), two Edit boxes (currently say Edit), a Horizontal Scroll bar, and a Group box. The Group box is the outlined rectangular area in the dialog box that encloses the other related controls. The OK and CANCEL buttons were added for you by the Resource Editor. You can delete (select the control and press DEL key), move (drag the control), resize (drag one of the anchor points), rename the OK and CANCEL buttons or any of the controls that you add. Dialog Editor Sample 2
Using Dialogs
Page 5 of 53
To specify the names and properties of the added controls: 1. Double-click one of the controls in your dialog box with the left mouse button. A Properties box appears showing the default name and properties for that control. The following figure shows the Properties box for the Horizontal Scroll bar with the default values. Dialog Editor Sample 3
2. Change the control name by typing over the default name (IDC_SCROLLBAR1 in the following figure). 3. Check or uncheck the available options to change the control's properties. (The Visible option in the following figure is checked by default.) 4. Click the left mouse button in the upper-right corner of the window Properties box to save the control's properties and to close the box. Repeat the same process for each control and for the dialog box itself. To use the controls from within a program, you need symbolic names for each of them. In this example, the Horizontal Scroll bar symbolic name is changed in the Properties box to IDC_SCROLLBAR_TEMPERATURE. This is how the control will be
Using Dialogs
Page 6 of 53
referred to in your program; for example, when you get the slide position:
INTEGER slide_position retlog = DLGGET (dlg, IDC_SCROLLBAR_TEMPERATURE, slide_position, DLG_POSITION) &
The top Edit box is named IDC_EDIT_CELSIUS. The Static text next to it is named IDC_TEXT_CELSIUS and set to the left-aligned text "Celsius". The lower Edit box is named IDC_EDIT_FAHRENHEIT, and the Static text next to it is named IDC_TEXT_FAHRENHEIT and set to the left-aligned text "Fahrenheit". The Group box is named IDC_BOX_TEMPERATURE, and its caption is set to Temperature. The dialog itself is named IDD_TEMP and its caption is set to Temperature Conversion. All other control properties are left at the default values.
The resulting dialog box is shown in the following figure: Dialog Editor Sample 4
To save the dialog box as a resource file: 1. From the File menu, choose Save As. 2. Enter a resource filename for your file. In this example, the resource file is given the name TEMP.RC. The visual development environment saves the resource file and creates an include
Using Dialogs
Page 7 of 53
file with the name RESOURCE.FD. Typically, only one resource file is used with each Visual Fortran project (see Including Resources Using Multiple Resource Files). 3. Add the TEMP.RC file to your project: 1. In the Project menu, select Add to Project and click Files... from the submenu. 2. The Insert Files into Project dialog box appears. Use this dialog box to select the TEMP.RC file to be added to the project. To open an existing dialog box in the resource editor: 1. 2. 3. 4. 5. 6. From the File menu, open the project workspace. Click the ResourceView tab. Click the plus sign (+) next to project-name Resources. Click the plus sign (+) next to Dialog. Double-click the appropriate dialog name, such as IDD_DIALOG1. Use the Resource Editor to add a new control or modify an existing control. To modify an existing control, use the Tab key to select the appropriate control. Double-click the selected control to view or modify its properties.
At this point the appearance of the dialog box is finished and the controls are named, but the box cannot function on its own. An application must be created to run it. Not all the controls on the Resource Editor Controls toolbar are supported by Visual Fortran dialog routines. The supported dialog controls are:
l l l l l l l l l l l l l l
Button Check box Combo box (such as a drop-down list box) Edit box Group box List box Picture Progress bar Radio button Scroll bar (Horizontal and Vertical) Slider Spin control Static text Tab control
You can also add ActiveX controls to your dialog box. For information, see Using ActiveX Controls. For further information on resources and control properties, see:
Using Dialogs
Page 8 of 53
l l l
Setting Control Properties Including Resources Using Multiple Resource Files The Include (.FD) File
For information on creating an application for your dialog box, see Writing a Dialog Application. For more information about using the Resource Dialog editor, see Using Resource Editors in the Visual C++ User's Guide (for Visual Fortran), located in the HTML Help folder Developer Studio 98.
In General tab of the Properties box, you can change the x and y values. These specify the pixel position of the dialog box's upper-left corner, relative to its parent window. For a modal or a popup modeless dialog box, the parent window is the screen. For example, specifying the X position as 40 and Y position as 40 would place a modal dialog box in the upper-left corner of the screen. In the More Styles tab, you can specify the Center checkbox to display the dialog box in the center of its parent window or specify the Center Mouse checkbox to place the dialog centered at the current mouse pointer position.
You can change the size of the dialog box by holding down the left mouse button as you drag the right or lower perimeter of the box. If you have sized your dialog window to be larger than the edit window, use the scroll bars to move the view region. You can edit the appearance of the dialog box later. To start the Resource Editor
Page 9 of 53
From the File menu, open the project workspace. Click the ResourceView tab. Click the plus sign (+) next to project-name Resources. Click the plus sign (+) next to Dialog. Double-click the appropriate dialog name, such as IDD_DIALOG1. Use the Resource Editor to modify an existing control or add a new one.
l l
To include resources that have already been developed and tested and do not need further modification. To include resources that are being used by several different projects, or that are part of a source code version-control system, and thus must exist in a central location where modifications will affect all projects. To include resources that are in a custom format. To include statements in your resource file that execute conditionally at compile time using compiler directives, such as #ifdef and #else. For example, your project may have a group of resources that are bracketed by #ifdef _DEBUG ... #endif and are thus included only if the constant _DEBUG is defined at compile time. To include statements in your resource file that modify resource-file syntax by using the #define directive to implement simple macros.
If you have sections in your existing .rc files that meet any of these conditions, you should place the sections in one or more separate .rc files and include them in your project using the Resource Includes dialog box. To include resource files that will be added to your project at compile time: 1. Place the resources in a resource script file with a unique filename. (Do not use projectname.rc, since this is the default filename used for the main resource script file.) 2. From the View menu, choose Resource Includes. 3. In the Compile-time directives box, use the #include compiler directive to include the new resource file in the main resource file in the development environment. The resources in files included in this way are made a part of your executable file at compile time. They are not directly available for
Using Dialogs
Page 10 of 53
editing or modification when you are working on your project's main .rc file. You need to open included .rc files separately. 4. Click OK.
Using Dialogs
Page 11 of 53
l l l l l l
Initializing and Activating the Dialog Box Callback Routines Using a Modeless Dialog Box Using Fortran AppWizards to Help Add Modal Dialog Box Coding Using Fortran AppWizards to Help Add Modeless Dialog Box Coding Using Dialog Controls in a DLL
This code associates the dialog type with the dialog (IDD_TEMP in this example) defined in your resource and include files (TEMP.RC and RESOURCE.FD in this example). You give your application access to your dialog resource by adding the .RC file to your project. You give your application access to the dialog include file by including the .FD file in each subprogram. You associate the dialog properties in these files with the dialog type by calling DLGINIT with your dialog name. An application that controls a dialog box should perform the following actions: 1. Call DLGINIT or DLGINITWITHRESOURCEHANDLE to initialize the dialog type and associate your dialog and its properties with the type. 2. Initialize the controls with the dialog set routines, such as DLGSET. 3. Set the callback routines to be executed when a user manipulates a control in the dialog box with DLGSETSUB. 4. Depending on whether you want a modal or modeless dialog type: To use a modal dialog, run the dialog with DLGMODAL. To use a modeless dialog, call DLGMODELESS and use DLGISDLGMESSAGE in your message loop. 5. Retrieve control information with the dialog get functions, such as DLGGET. 6. Free resources from the dialog with DLGUNINIT. As an example of activating a dialog box and controls, the following code initializes the temperature dialog box and controls created in the TEMP project example. It also sets the callback routine as UpdateTemp, displays the dialog box, and releases the dialog resources when done:
Using Dialogs
Page 12 of 53
SUBROUTINE DoDialog( ) USE DFLOGM IMPLICIT NONE INCLUDE 'RESOURCE.FD' INTEGER retint LOGICAL retlog TYPE (dialog) dlg EXTERNAL UpdateTemp Initialize. IF ( .not. DlgInit( idd_temp, dlg ) ) THEN WRITE (*,*) "Error: dialog not found" ELSE Set up temperature controls. retlog = DlgSet( dlg, IDC_SCROLLBAR_TEMPERATURE, 200, DLG_RANGEMAX) retlog = DlgSet( dlg, IDC_EDIT_CELSIUS, "100" ) CALL UpdateTemp( dlg, IDC_EDIT_CELSIUS, DLG_CHANGE) retlog = DlgSetSub( dlg, IDC_EDIT_CELSIUS, UpdateTemp ) retlog = DlgSetSub( dlg, IDC_EDIT_FAHRENHEIT, UpdateTemp ) retlog = DlgSetSub( dlg, IDC_SCROLLBAR_TEMPERATURE, UpdateTemp ) Activate the modal dialog. retint = DlgModal( dlg ) Release dialog resources. CALL DlgUninit( dlg ) END IF END SUBROUTINE DoDialog
! !
The dialog routines, such as DLGSET and DLGSETSUB, refer to the dialog controls by the names you assigned to them in the Properties box while creating the dialog box in the Resource Editor. For example:
retlog = DlgSet( dlg, IDC_SCROLLBAR_TEMPERATURE, 200, DLG_RANGEMAX)
In this statement, the dialog function DLGSET assigns the control named IDC_SCROLLBAR_TEMPERATURE a value of 200. The index DLG_RANGEMAX specifies that this value is a scroll bar maximum range. Consider the following:
retlog = DlgSet( dlg, IDC_EDIT_CELSIUS, "100" ) CALL UpdateTemp( dlg, IDC_EDIT_CELSIUS, DLG_CHANGE)
The preceding statements set the dialog's top Edit box, named IDC_EDIT_CELSIUS in the Resource Editor, to an initial value of 100, and calls the routine UpdateTemp to inform the application that the value has changed. Consider the following:
retlog = DlgSetSub( dlg, IDC_EDIT_CELSIUS, UpdateTemp ) retlog = DlgSetSub( dlg, IDC_EDIT_FAHRENHEIT, UpdateTemp ) retlog = DlgSetSub( dlg, IDC_SCROLLBAR_TEMPERATURE,UpdateTemp )
The preceding statements associate the callback routine UpdateTemp with the three controls. This results in the UpdateTemp routine being called whenever the value of any of the three controls changes. Routines are assigned to the controls with the function DLGSETSUB. Its first argument is the dialog variable, the second is the control name, the third is the
Using Dialogs
Page 13 of 53
name of the routine you have written for the control, and the optional fourth argument is an index to select between multiple routines. You can set the callback routines for your dialog controls anywhere in your application: before opening your dialog with either DLGMODAL or DLGMODELESS, or from within another callback routine. In the TEMP example, the main program calls the DoDialog subroutine to display the Temperature Conversion dialog box.
Using Dialogs
local_callbacktype = callbacktype
Page 14 of 53
SELECT CASE (control_name) CASE (IDC_EDIT_CELSIUS) ! Celsius value was modified by the user so ! update both Fahrenheit and Scroll bar values. retlog = DlgGet( dlg, IDC_EDIT_CELSIUS, text ) READ (text, *, iostat=retint) cel IF ( retint .eq. 0 ) THEN far = (cel-0.0)*((212.0-32.0)/100.0)+32.0 WRITE (text,*) far retlog = DlgSet( dlg, IDC_EDIT_FAHRENHEIT, & & TRIM(ADJUSTL(text)) ) retlog = DlgSet( dlg, IDC_SCROLLBAR_TEMPERATURE, cel, & & DLG_POSITION ) END IF CASE (IDC_EDIT_FAHRENHEIT) ! Fahrenheit value was modified by the user so ! update both celsius and Scroll bar values. retlog = DlgGet( dlg, IDC_EDIT_FAHRENHEIT, text ) READ (text, *, iostat=retint) far IF ( retint .eq. 0 ) THEN cel = (far-32.0)*(100.0/(212.0-32.0))+0.0 WRITE (text,*) cel retlog = DlgSet( dlg, IDC_EDIT_CELSIUS, TRIM(ADJUSTL(text)) ) retlog = DlgSet( dlg, IDC_SCROLLBAR_TEMPERATURE, cel, & & DLG_POSITION ) END IF CASE (IDC_SCROLLBAR_TEMPERATURE) ! Scroll bar value was modified by the user so ! update both Celsius and Fahrenheit values. retlog = DlgGet( dlg, IDC_SCROLLBAR_TEMPERATURE, cel, & & DLG_POSITION ) far = (cel-0.0)*((212.0-32.0)/100.0)+32.0 WRITE (text,*) far retlog = DlgSet( dlg, IDC_EDIT_FAHRENHEIT, TRIM(ADJUSTL(text)) ) WRITE (text,*) cel retlog = DlgSet( dlg, IDC_EDIT_CELSIUS, TRIM(ADJUSTL(text)) ) END SELECT END SUBROUTINE UpdateTemp
Each control in a dialog box, except a pushbutton, has a default callback that performs no action. The default callback for a pushbutton's click event sets the return value of the dialog to the pushbutton's name and then exits the dialog. This makes all pushbuttons exit the dialog by default, and gives the OK and CANCEL buttons good default behavior. A routine that calls DLGMODAL can then test to see which pushbutton caused the modal dialog to exit. Callbacks for a particular control are called after the value of the control has been changed by the user's action. Calling DLGSET does not cause a callback to be called for the changing value of a control. In particular, when inside a callback, performing a DLGSET on a control will not cause the associated callback for that control to be called. Calling DLGSET before or after DLGMODAL or DLGMODELESS has been called also does not cause the callback to be called. If the callback needs to be called, it can be called manually using CALL after the DLGSET is performed.
Using Dialogs
Page 15 of 53
! Include the constants provided by the Resource Editor include 'resource.fd' ! A dialog box callback external ThermometerSub ! Variables type (dialog) dlg type (T_MSG) mesg integer(4) ret logical(4) lret ! Create the thermometer dialog box and set up the controls and callbacks
Using Dialogs
lret lret lret lret lret lret = = = = = =
Page 16 of 53
DlgInit(IDD_THERMOMETER, dlg_thermometer) DlgSetSub(dlg_thermometer, IDD_THERMOMETER, ThermometerSub) DlgSet(dlg_thermometer, IDC_PROGRESS1, 32, DLG_RANGEMIN) DlgSet(dlg_thermometer, IDC_PROGRESS1, 212, DLG_RANGEMAX) DlgSet(dlg_thermometer, IDC_PROGRESS1, 32) DlgModeless(dlg_thermometer, nCmdShow)
! Read and process messages until GetMessage returns 0 because ! PostQuitMessage has been called do while( GetMessage (mesg, NULL, 0, 0) ) ! Note that DlgIsDlgMessage must be called in order to give ! the dialog box first chance at the message. if ( DlgIsDlgMessage(mesg) .EQV. .FALSE. ) then lret = TranslateMessage( mesg ) ret = DispatchMessage( mesg ) end if end do ! Cleanup dialog box memory call DlgUninit(dlg) ! The return value is the wParam of the Quit message WinMain = mesg.wParam return end function
Using Dialogs
HelloDlg.rc
4. In the Project menu, select Add To Project -> Files. Select HelloDlg.rc and click OK. 5. Edit (double-click its name in the FileView ) the file HelloDlg.f90: After the program HELLODLG line, add the following line:
USE DFLOGM
The USE DFLOGM line includes the DFLOGM module interfaces to the Dialog routines. The line include 'resource.fd' includes the .fd file. The function reference to DLGINIT initializes the dialog box. The function reference to DLGMODAL displays the dialog box. The call to DLGUNINIT frees the dialog box resources.
6. Build the Hellodlg Fortran Console project application. When you execute the application, the dialog box you created appears in front of the Console window:
Using Dialogs
Page 18 of 53
7. Click OK to dismiss the dialog box. For Visual Fortran applications using the Fortran Windows project type, you can use the Fortran Windows Project AppWizard to help you add dialog coding for a modeless dialog box. For information about coding requirements for modal and modeless dialog boxes, see Initializing and Activating the Dialog Box.
A simple Dialog Based Application To create an application where a dialog box is the main window of the application, without a menu bar, choose the A simple Dialog Based Application option. This creates the skeleton of an entire application that you can immediately build and run to display a sample dialog box. You can add controls to the dialog box and add dialog procedure calls to manipulate the controls and handle dialog callbacks. A number of the sample projects in the ...\SAMPLES\DIALOG\ directory were started using the A simple Dialog Based Application option.
A simple Single Document Interface (SDI) To create an application where a dialog box is the main window of the application with a menu bar, choose the A simple Single Document Interface (SDI) option.
Using Dialogs
Page 19 of 53
This also creates the skeleton of an entire application that you can immediately build and run. You can add a dialog box to the client area of the main window (as explained later). In the template-like code generated when you select A simple Dialog Based Application option:
l
After you create a project with the A simple Dialog Based Application option, build it, and execute it, the following dialog box is displayed:
Some of the code specific to the dialog routine interfaces and data declarations follows. For this example, the project name is FWin. The project name is used in some of the data declarations:
use dflogm use FWin_dialogGlobals . . . include 'resource.fd' external FWin_dialogSub external FWin_dialogApply ! Variables type (T_MSG) integer*4 logical*4 mesg ret lret
The FWin_dialogGlobals module is defined in a separate source file in that project. The FWin_dialogSub and FWin_dialogApply are subroutines defined later in the main source file that are callback routines for different controls for the dialog box.
l
Using Dialogs
if (lret == .FALSE.) goto 99999
Page 20 of 53
Notes for this example: DlgInit initializes the dialog box. The first call to DlgSetSub assigns a callback subroutine to the Exit button. It associates the FWin_dialogSub subroutine with the dialog box identifier IDD_FWIN_DIALOG_DIALOG (project name is FWin_Dialog). The FWin_dialogSub routine contains code to terminate the program. The second call to DlgSetSub associates FWin_dialogApply with the Apply button identifier IDM_APPLY. The user should add code in the FWin_dialogApply subroutine to take appropriate action. DlgModeless displays the initialized modeless dialog box, which is ready for user input.
l
The code specific to processing messages (message loop) to react to user input follows:
! Read and process messages do while( GetMessage (mesg, NULL, 0, 0) ) if ( DlgIsDlgMessage(mesg) .EQV. .FALSE. ) then lret = TranslateMessage( mesg ) ret = DispatchMessage( mesg ) end if end do
Notes for this example: The GetMessage Win32 call inside a DO WHILE loop returns a message from the calling thread's message queue. DlgIsDlgMessage determines whether the specified message is intended for one of the currently displayed modeless dialog boxes, or a specific dialog box. The TranslateMessage Win32 call translates virtual-key messages into character messages. The DispatchMessage Win32 call dispatches a message to a window procedure.
l
The dialog box is terminated and its resources are released by calling DlgUninit:
call DlgUninit(gdlg)
Using Dialogs
Page 21 of 53
In the template-like code generated when you select A simple Single Document Interface (SDI) AppWizard option, to add the dialog box to the client area of the main window: 1. In the Insert menu, select Resource... and create a new Dialog box. Edit the Dialog Properties. Select the Styles tab and set Styles to Child and Border to Thin. 2. In the main source file, add the following USE statement:
USE dflogm
3. In the main source file, in the function MainWndProc, add a case to handle the WM_CREATE message. In this case, initialize the dialog box in the normal manner. To display the dialog box, call:
lret = DlgModeless(dlg, SW_SHOWNA, hwndParent)
In this call, hwndParent is the window handle of the application's main window. 4. In the main source file, add a call to DlgIsDlgMessage to the message loop, before the call to the Win32 routine TranslateAccelerator. It should look like:
! Read and process messages do while( GetMessage (mesg, NULL, 0, 0) ) if ( DlgIsDlgMessage(mesg, dlg) .EQV. .FALSE. ) then if ( TranslateAccelerator (mesg%hwnd, haccel, mesg) == 0) then lret = TranslateMessage( mesg ) ret = DispatchMessage( mesg ) end if end if end do
5. Optionally, if you want to allow the user to resize the main window, add a case to handle the WM_RESIZE message and change the layout of the dialog box based upon its size. See the FXPLORER Visual Fortran Sample in ...\SAMPLES\DIALOG for an example of this type of application.
Using Dialogs
Page 22 of 53
To do this, initialize your dialog box using DlgInitWithResourceHandle rather than DlgInit. As compared to DlgInit, DlgInitWithResourceHandle takes an additional argument named "hinst". The "hinst" argument is the module instance handle in which the dialog resource can be found. For a DLL, this handle is passed into the DLL entry point, DllMain. An example of a DllMain function follows:
module dll_globals integer(4) ghInst end module dll_globals ! DLL instance handle
!******************************************************************** !* FUNCTION: DllMain(HANDLE, DWORD, LPVOID) !* !* PURPOSE: DllMain is called by Windows when !* the DLL is initialized, Thread Attached, and other times. !* Refer to SDK documentation, as to the different ways this !* may be called. !* !* The DllMain function should perform additional initialization !* tasks required by the DLL. DllMain should return a value of 1 !* if the initialization is successful. !* !********************************************************************* integer(4) function DllMain (hInst, ul_reason_being_called, lpReserved) !DEC$ IF DEFINED(_X86_) !DEC$ ATTRIBUTES STDCALL, ALIAS : '_DllMain@12' :: DllMain !DEC$ ELSE !DEC$ ATTRIBUTES STDCALL, ALIAS : 'DllMain' :: DllMain !DEC$ ENDIF use dll_globals integer(4) hInst integer(4) ul_reason_being_called integer(4) lpReserved ! Save the module instance handle in a global variable ! This would typically be in a Module or a COMMON block. ghInst = hInst DllMain = 1 return end
One way to use DlgInitWithResourceHandle is to build a resource-only DLL. A resource-only DLL contains an .RC file, but no code. It is useful for building an application that supports multiple languages. You can create a main application and several resource-only DLLs (one for each language) and call the Win32 LoadLibrary routine at the beginning of your application to load the appropriate resource-only DLL. To use a dialog box from the resource-only DLL, first call LoadLibrary (see the Platform SDK online documentation) to return the instance handle that you can use when you call DlgInitWithResourceHandle.
Using Dialogs
Page 23 of 53
When you create a Fortran DLL project, you can create a resource-only DLL using the Fortran Dynamic Link Library AppWizard: 1. In the Fortran Dynamic Link Library AppWizard, select "An empty DLL application." 2. Complete creating the project. 3. In the Project menu, select Add to Project...Files to add your .RC file and the RESOURCE.H file that defines the identifiers of the controls. 4. In the Project menu: Click Settings Click Link tab Add the /noentry switch to the end of the options in the Project Options edit box
Dialog Routines
You can use dialog routines as you would any intrinsic procedure or run-time routine. As described in Using Dialogs, Visual Fortran supports two types of dialog boxes: modal and modeless. You can use a modal dialog box with any Fortran project type. You can use a modeless dialog box only with the Fortran Windows project types. The dialog routines can:
l l l l
Initialize and close the dialog box Retrieve user input from a dialog box Display data in the dialog box Modify the dialog box controls
The include file (.FD) of the dialog box contains the names of the dialog controls that you specified in the Properties box of the Resource Editor when you created the dialog box. The module DFLOGM.MOD contains predefined variable names and type definitions. These control names, variables, and type definitions are used in the dialog routine argument lists to manage your dialog box. The dialog routines are listed in the following table: Dialog Routine DLGEXIT DLGFLUSH Description Closes an open dialog Updates the dialog display
Using Dialogs DLGGET DLGGETCHAR DLGGETINT DLGGETLOG DLGINIT DLGINITWITHRESOURCEHANDLE DLGISDLGMESSAGE DLGISDLGMESSAGEWITHDLG
Page 24 of 53 Gets the value of a control variable Gets the value of a character control variable Gets the value of an integer control variable Gets the value of a logical control variable Initializes the dialog Initializes the dialog (alternative to DLGINIT) Determines whether a message is intended for a modeless dialog box Determines whether a message is intended for a modeless dialog box (alternative to DLGISDLGMESSAGE) Displays a modal dialog box Displays a modal dialog box (alternative to DLGMODAL) Displays a modeless dialog box Sends a message to a control Assigns a value to a control variable Assigns a value to a character control variable Assigns a routine to handle an ActiveX control event. Assigns a value to an integer control variable Assigns a value to a logical control variable Sets the return value for DLGMODAL Assigns a defined callback routine to a control
DLGMODAL DLGMODALWITHPARENT DLGMODELESS DLGSENDCTRLMESSAGE DLGSET DLGSETCHAR DLGSETCTRLEVENTHANDLER DLGSETINT DLGSETLOG DLGSETRETURN DLGSETSUB
Page 25 of 53
These routines are described in the Language Reference (see also Dialog Procedures: table).
Dialog Controls
Each dialog control in a dialog box has a unique integer identifier and name. You specify the name in the Properties box for each control within the Resource Editor, and the Resource Editor assigns the PARAMETER attribute and an integer value to each control name. You can refer to a control by its name, for example IDC_SCROLLBAR_TEMPERATURE, or by its integer value, which you can read from the include (.FD) file. Each dialog control has one or more variables associated with it, called control indexes. These indexes can be integer, logical, character, or external. For example, a plain Button has three associated variables: one is a logical value associated with its current enabled state, one is a character variable that determines its title, and the third is an external variable that indicates the subroutine to be called if a mouse click occurs. Dialog controls can have multiple variables of the same type. For example, the scroll bar control has four integer variables associated with it:
l l l l
Scroll bar position Scroll bar minimum range Scroll bar maximum range Position change if the user clicks on the scroll bar space next to the slide (big step)
Control Indexes Available Indexes for Each Dialog Control Specifying Control Indexes
Control Indexes
The value of a dialog control's index is set with the DLGSET functions: DLGSET, DLGSETINT, DLGSETLOG, DLGSETCHAR, and DLGSETSUB. The control name and control index name are arguments to the DLGSET functions and specify the particular control index being set. For example:
Using Dialogs
Page 26 of 53
The index DLG_POSITION specifies the scroll bar position is set to 45. Consider the following:
retlog = DlgSet( dlg, IDC_SCROLLBAR_TEMPERATURE, 200, DLG_RANGEMAX)
In this statement, the index DLG_RANGEMAX specifies the scroll bar maximum range is set to 200. The DLGSET functions have the following syntax: result = DLGSET (dlg, control_name, value, control_index_name) The control_index_name determines what the value in the DLGSET function means. The control index names are declared in the module DFLOGM.MOD and should not be declared in your routines. Available control indexes and how they specify the interpretation of the value argument are listed in the following Control Indexes table. Control Index DLG_ADDSTRING DLG_BIGSTEP How the Value is Interpreted Used with DLGSETCHAR to add an entry to a List box or Combo box The amount of change that occurs in a Scroll bar's or Slider's position when the user clicks beside the Scroll bar's or slider's slide (default = 10) A subroutine called after the user has modified a control and the control has been updated on the screen A subroutine called when the control receives a mouseclick A subroutine called when a control is double-clicked Same as not specifying a control index The enable state of the control (value = .TRUE. means enabled, value = .FALSE. means disabled) A subroutine called when an Edit Box receives input focus. The object pointer of an ActiveX control.
Page 27 of 53 A subroutine called when an Edit Box loses input focus. The total number of items in a List box, Combo box, or Tab control The current position of the Scroll bar, Spin, Slider, or Progress bar. Also, the current cursor position in the edit box. The minimum value of a Scroll bar's, Spin's, Slider's, or Progress' position (default = 1 for scroll bar, 0 for other controls) The maximum value of a Scroll bar's, Spin's, Slider's, or Progress' position (default = 100) A subroutine called when the selection in a List Box or Combo Box changes A subroutine called when the selected Tab control is about to be changed. In this subroutine, calling DLGGETINT with the index DLG_STATE refers to the Tab that was active before the change. The amount of change that occurs in a Slider's position when the user presses the keyboard arrow keys (default = 1) The user changeable state of a control The length of text in an edit box. The interval frequency for tick marks in a Slider (default = 1) The title text associated with a control A subroutine called after the user has modified the control state but before the control has been updated on the screen
DLG_RANGEMIN
DLG_SMALLSTEP
The index names associated with dialog controls do not need to be used unless there is more than one variable of the same type for the control and you do not want the default variable. For example:
retlog = DlgSet(dlg, IDC_SCROLLBAR_TEMPERATURE, 45, DLG_POSITION) retlog = DlgSet(dlg, IDC_SCROLLBAR_TEMPERATURE, 45)
Using Dialogs
Page 28 of 53
These statements both set the Scroll bar position to 45, because DLG_POSITION is the default control index for the scroll bar. Dialog Indexes The control identifier specified in DLGSETSUB can also be the identifier of the dialog box. In this case, the index must be one of the values listed in the Dialog Indexes table: Dialog Index DLG_INIT How the Value is Interpreted A subroutine called after the dialog box is created but before it is displayed (with callbacktype=DLG_INIT) and immediately before the dialog box is destroyed (with callbacktype=DLG_DESTROY). A subroutine called after the dialog box is resized.
DLG_SIZECHANGE
For more information on dialog controls, see Available Indexes for Each Dialog Control.
Using Dialogs Combo box DLG_NUMITEMS Sets or returns the total number of items in a list DLG_ENABLE Use DLG_STATE, DLG_ADDSTRING, or an index: DLG_STATE By default, sets or returns the text of the selected item or first item in the list DLG_ADDSTRING Used with DLGSETCHAR to add a new item. It automatically increments DLG_NUMITEMS. An index, 1 to n Sets or returns the text of a particular item Dropdown list box Use DLG_NUMITEMS or DLG_STATE: DLG_NUMITEMS (default) Sets or returns the total number of items in a list DLG_STATE Sets or returns the index of the selected item DLG_ENABLE Use DLG_STATE, DLG_ADDSTRING, or an index: DLG_STATE By default, sets or returns the text of the selected item or first item in the list, or you can include an index, 1 to n, to set or return indicates the text of a particular item DLG_ADDSTRING Used with DLGSETCHAR to add a new item. It automatically increments
Using Dialogs DLG_NUMITEMS. Edit box DLG_TEXTLENGTH (default) Sets or returns the length of the text in the edit box. DLG_POSITION Sets or returns the cursor position Group box List box Use DLG_NUMITEMS or an index: DLG_NUMITEMS Sets or returns the total number of items in a list An index, 1 to n Determines which list items have been selected and their order DLG_ENABLE DLG_ENABLE DLG_TITLE Use DLG_STATE, DLG_ADDSTRING, or an index: DLG_STATE By default, returns the text of the first selected item DLG_ADDSTRING Used with DLGSETCHAR to add a new item. It automatically increments DLG_NUMITEMS. An index, 1 to n Sets or returns the text of a particular item Picture Progress bar DLG_POSITION (default) DLG_RANGEMIN DLG_RANGEMAX DLG_ENABLE DLG_ENABLE DLG_ENABLE DLG_STATE
Page 30 of 53
Using Dialogs Radio button Scroll bar DLG_POSITION (default) DLG_RANGEMIN DLG_RANGEMAX DLG_BIGSTEP DLG_POSITION (default) DLG_RANGEMIN DLG_RANGEMAX DLG_SMALLSTEP DLG_BIGSTEP DLG_TICKFREQ DLG_POSITION (default) DLG_RANGEMIN DLG_RANGEMAX DLG_STATE (default) DLG_ENABLE DLG_ENABLE DLG_TITLE
Page 31 of 53 DLG_CLICKED
DLG_CHANGE
Slider
DLG_ENABLE
DLG_CHANGE
Spin controls
DLG_ENABLE
DLG_CHANGE
Static text Tab control Use DLG_NUMITEMS (default), DLG_STATE, or an index: DLG_NUMITEMS Sets or returns the total number of tabs DLG_STATE Sets or returns the currently selected tab An index, 1 to n Sets or returns the dialog name of the dialog box associated with a
DLG_ENABLE DLG_ENABLE
DLG_TITLE Use DLG_STATE or an index: DLG_STATE By default, sets or returns the currently selected tab An index, 1 to n Sets or returns the text of a particular Tab DLG_SELCHANGE (default) DLG_SELCHANGING
Using Dialogs particular tab For an overview on control indexes, see Control Indexes.
Page 32 of 53
You do not need the control index DLG_TITLE because there is only one character index for a Static text control. The generic function DLGSET chooses the control index to change based on the argument type, in this case CHARACTER. For each type of index, you can use the generic DLGSET function or the specific DLGSET function for that type: DLGSETINT, DLGSETLOG, or DLGSETCHAR. For example, you can disable the Static text control IDC_TEXT_CELSIUS by setting its logical value to .FALSE. with either DLGSET or DLGSETLOG:
retlog = DLGSETLOG (dlg, IDC_TEXT_CELSIUS, .FALSE., DLG_ENABLE) retlog = DLGSET (dlg, IDC_TEXT_CELSIUS, .FALSE., DLG_ENABLE)
In both these cases, the control index DLG_ENABLE can be omitted because there is only one logical control index for Static text controls. You can query the value of a particular control index with the DLGGET functions, DLGGET, DLGGETINT, DLGGETLOG, and DLGGETCHAR. For example:
INTEGER current_val LOGICAL are_you_enabled retlog = DLGGET (dlg, IDC_SCROLLBAR_TEMPERATURE, current_val, & DLG_RANGEMAX) retlog = DLGGET (dlg, IDC_SCROLLBAR_TEMPERATURE, are_you_enabled, & DLG_ENABLE)
& &
This code returns the maximum range and the enable state of the scroll bar. The arguments you declare (current_val and are_you_enabled in the preceding example) to hold the queried values must be of the same type as the values retrieved. If you use specific DLGGET functions such as DLGGETINT or DLGGETCHAR, the control index value retrieved must be the appropriate type. For example, you cannot use DLGGETCHAR to retrieve an integer or logical value. The DLGGET functions return .FALSE. for illegal type combinations. You cannot query for the name of an external callback routine.
Using Dialogs
Page 33 of 53
In general, it is better to use the generic functions DLGSET and DLGGET rather than their type-specific variations because then you do not have to worry about matching the function to type of value set or retrieved. DLGSET and DLGGET perform the correct operation automatically, based on the type of argument you pass to them. More information on these routines is available in the Language Reference.
This example disables the control named IDC_CHECKBOX1. When you create your dialog box in the Resource Editor, the dialog controls are given a tab order. When the user hits the Tab key, the dialog box focus shifts to the next control in the tab order. By default, the tab order of the controls follows the order in which they were created. This may not be the order you want. You can change the order by opening the Layout menu and choosing Tab Order (or by pressing the key combination Ctrl+D) in the Resource Editor. A tab number will appear next to each control. Click the mouse on the control you want to be first, then on the control you want to be second in the tab order and so on. Tab order also determines which control gets the focus if the user presses the Group box hotkey. (See Using Group Boxes.) For information on Visual Fortran Samples that use the Dialog functions, see the ...\DF98\SAMPLES\DIALOG folder. The following sections describe the function and use of the dialog controls:
l l l l l
Static Text Edit Boxes Group Boxes Check Boxes and Radio Buttons Buttons
Using Dialogs
l l l l l l l l
Page 34 of 53
Using List Boxes and Combo Boxes Using Scroll Bars Using Pictures Using Progress Bars Using Spin Controls Using Sliders Using Tab Controls Setting Return Values and Exiting
For information on using ActiveX controls in a dialog, see Using ActiveX Controls.
The values a user enters into the Edit box are always retrieved as character strings, and your application needs to interpret these strings as the data they represent. For example, numbers entered by the user are interpreted by your application as character strings. Likewise, numbers you write to the Edit box are sent as character strings. You can convert between numbers and strings by using internal read and write statements to make type conversions. To read a number in the Edit box, retrieve it as a character string with DLGGET or DLGGETCHAR, and then execute an internal read using a variable of the numeric type you want (such as integer or real). For example:
REAL x LOGICAL retlog CHARACTER(256) text retlog = DLGGET (dlg, IDC_EDITBOX1, text) READ (text, *) x
Using Dialogs
Page 35 of 53
In this example, the real variable x is assigned the value that was entered into the Edit box, including any decimal fraction. Complex and double complex values are read the same way, except that your application must separate the Edit box character string into the real part and imaginary part. You can do this with two separate Edit boxes, one for the real and one for the imaginary part, or by requiring the user to enter a separator between the two parts and parsing the string for the separator before converting. If the separator is a comma (,) you can read the string with two real edit descriptors without having to parse the string. To write numbers to an Edit box, do an internal write to a string, then send the string to the Edit box with DLGSET. For example:
INTEGER j LOGICAL retlog CHARACTER(256) text WRITE (text,'(I4)') j retlog = DLGSET (dlg, IDC_EDITBOX1, text)
Use the DLG_TEXTLENGTH control index to get or set the length of the characters in the edit box. The length is automatically updated when:
l
Your program calls DLGSET to set the text in the edit box (trailing blanks are stripped before setting the edit box). The user modifies the text in the edit box.
If you want to set the text with significant trailing blanks, call DLGSET to set the text followed by DLGSET with the DLG_TEXTLENGTH index to set the length that you want. Use the DLG_POSITION index to get or set the current cursor position in the edit box. Note that setting the cursor position cancels any current selection in the edit box.
This causes the "T" to be underlined in the title and makes it a hotkey. When the user presses the key combination ALT+T, the focus of the dialog box shifts to
Using Dialogs
Page 36 of 53
the next control after the Group box in the tab order. This control should be a control in the group. (You can view and change the tab order from the Layout/Tab Order menu option in the Resource Editor.) Disabling the Group box disables the hotkey, but does not disable any of the controls within the group. As a matter of style, you should generally disable the controls in a group when you disable the Group box.
If you need to change the state of the button, for initialization or in response to other user input, you use DLGSET or DLGSETLOG. For example:
LOGICAL retlog retlog = DLGSET (dlg, IDC_RADIOBUTTON1, .TRUE.) retlog = DLGSET (dlg, IDC_CHECKBOX1, .TRUE.)
Radio buttons are typically used in a group where the user can select only one of a set of radio buttons. When using Radio buttons with the Dialog routines, use the following guidelines:
l
Each Radio button should have the "Auto" style set. This is the default for a new Radio button. The first Radio button in a group should have the "Group" style set. This is not the default for a new Radio button. The remaining Radio buttons in the group should not have the "Group" style set, and should immediately follow the first button in the dialog box "Tab order". The default tab order is the order in which you create the controls. You can view and change the tab order from the Layout/Tab Order menu option in the Resource Editor. When the user selects a Radio button in a group, its state is set to .TRUE. and the state of the other Radio buttons in the group is set to .FALSE.. To set the currently selected Radio button from your code, call DLGSETLOG to set the selected Radio button to .TRUE.. Do not set the other Radio buttons to .FALSE.. This is handled automatically.
Using Buttons
Using Dialogs
Page 37 of 53
Unlike Check Boxes and Radio Buttons, Buttons do not have a state. They do not hold the value of being pushed or not pushed. When the user clicks on a Button with the mouse, the Button's callback routine is called. Thus,the purpose of a Button is to initiate an action. The external procedure you assign as a callback determines the action initiated. For example:
LOGICAL retlog EXTERNAL DisplayTime retlog = DlgSetSub( dlg, IDC_BUTTON_TIME, DisplayTime)
Using List boxes Using Combo boxes Using Drop-down List boxes
Using Dialogs
Page 38 of 53
For both List boxes and Combo boxes, the control index DLG_NUMITEMS determines how many items are in the box. Once this value is set, you set the text of List box items by specifying a character string for each item index. Indexes run from 1 to the total number of list items set with DLG_NUMITEMS. For example:
LOGICAL retlog retlog = DlgSet retlog = DlgSet retlog = DlgSet retlog = DlgSet ( ( ( ( dlg, dlg, dlg, dlg, IDC_LISTBOX1, IDC_LISTBOX1, IDC_LISTBOX1, IDC_LISTBOX1, 3, DLG_NUMITEMS ) "Moe", 1 ) "Larry", 2 ) "Curly", 3 )
These function calls to DLGSET put three items in the List box. The initial value of each List box entry is a blank string and the value becomes nonblank after it has been set. You can change the list length and item values at any time, including from within callback routines. If the list is shortened, the set of entries is truncated. If the list is lengthened, blank entries are added. In the preceding example, you could extend the list length and define the new item with the following:
retlog = DLGSET ( dlg, IDC_LISTBOX1, 4) retlog = DLGSET ( dlg, IDC_LISTBOX1, "Shemp", 4)
Since List boxes allow selection of multiple entries, you need a way to determine which entries are selected. When the user selects a List box item, it is assigned an integer index. You can test which list items are selected by reading the selection indexes in order until a zero value is read. For example, if in the previous List box the user selected Moe and Curly, the List box selection indexes would have the following values: Selection Index 1 2 3 Value 1 (for Moe) 3 (for Curly) 0 (no more selections)
If Larry alone had been selected, the List box selection index values would be: Selection Index 1 2 Value 2 (for Larry) 0 (no more selections)
Using Dialogs
Page 39 of 53
To determine the items selected, the List box values can be read with DLGGET until a zero is encountered. For example:
INTEGER j, num, test INTEGER, ALLOCATABLE :: values(:) LOGICAL retlog retlog = DLGGET (dlg, IDC_LISTBOX1, num, DLG_NUMITEMS) ALLOCATE (values(num)) j = 1 test = -1 DO WHILE (test .NE. 0) retlog = DLGGET (dlg, IDC_LISTBOX1, values(j), j) test = values(j) j = j + 1 END DO
In this example, j is the selection index and values(j) holds the list numbers, of the items selected by the user, if any. To read a single selection, or the first selected item in a set, you can use DLG_STATE, since for a List Box DLG_STATE holds the character string of the first selected item (if any). For example:
! Get the string for the first selected item. retlog = DLGGET (dlg, IDC_LISTBOX1, str, DLG_STATE)
Alternatively, you can first retrieve the list number of the selected item, and then get the string associated with that item:
INTEGER value CHARACTER(256) str ! Get the list number of the first selected item. retlog = DLGGET (dlg, IDC_LISTBOX1, value, 1) ! Get the string for that item. retlog = DLGGET (dlg, IDC_LISTBOX1, str, value)
In these examples, if no selection has been made by the user, str will be a blank string. In the Properties/Styles box in the Resource Editor, List boxes can be specified as sorted or unsorted. The default is sorted, which causes List box items to be sorted alphabetically starting with A. If a List box is specified as sorted, the items in the list are sorted whenever they are updated on the screen. This occurs when the dialog box is first displayed and when the items are changed in a callback. The alphabetical sorting follows the ASCII collating sequence, and uppercase letters come before lowercase letters. For example, if the List box in the example above with the list "Moe," "Larry," "Curly," and "Shemp" were sorted, before a callback or after DLGMODAL returned, index 1 would refer to "Curly,"
Using Dialogs
Page 40 of 53
index 2 to "Larry," index 3 to "Moe," and index 4 to "Shemp." For this reason, when using sorted List boxes, indexes should not be counted on to be the same once the dialog is displayed and any change is made to the list items. You can also call DLGSETCHAR with the DLG_ADDSTRING index to add items to a List box or Combo box. For example:
retlog = DlgSet(dlgtab, IDC_LIST, "Item 1", DLG_ADDSTRING)
When you use DLG_ADDSTRING, the DLG_NUMITEMS control index of the List or Combo box is automatically incremented. When adding items to a sorted list or Combo box, using DLG_ADDSTRING can be much easier than the alternative (setting DLG_NUMITEMS and then setting items using an index value), because you need not worry about the list being sorted and the index values changing between calls. Using Combo Boxes A Combo box is a combination of a List box and an Edit box. The user can make a selection from the list that is then displayed in the Edit box part of the control, or enter text directly into the Edit box. All dialog values a user enters are character strings, and your application must interpret these strings as the data they represent. For example, numbers entered by the user are returned to your application as character strings. Because user input can be given in two ways, selection from the List box portion or typing into the Edit box portion directly, you need to register two callback types with DLGSETSUB for a Combo box. These callback types are dlg_selchange to handle a new list selection by the user, and dlg_update to handle text entered by the user directly into the Edit box portion. For example:
retlog = DlgSetSub( dlg, IDC_COMBO1, UpdateCombo, dlg_selchange ) retlog = DlgSetSub( dlg, IDC_COMBO1, UpdateCombo, dlg_update )
A Combo box list is created the same way a List box list is created, as described in the previous section, but the user can select only one item from a Combo box at a time. When the user selects an item from the list, Windows automatically puts the item into the Edit box portion of the Combo box. Thus, there is no need, and no mechanism, to retrieve the item list number of a selected item. If the user is typing an entry directly into the Edit box part of the Combo box, again Windows automatically displays it and you do not need to. You can retrieve the character string of the selected item or Edit box entry with the following statement:
! Returns the character string of the selected item or Edit box entry as str.
Using Dialogs
retlog = DLGGET (dlg, IDC_COMBO1, str)
Page 41 of 53
Like List boxes, Combo boxes can be specified as sorted or unsorted. The notes about sorted List boxes also apply to sorted Combo boxes. You have three choices for Combo box Type in the Styles tab of Combo box Properties:
l l l
Simple and Drop-down are the same, except that a simple Combo box always displays the Combo box choices in a list, while a Drop-down list Combo box has a Drop-down button and displays the choices in a Drop-down list, conserving screen space. The Drop list type is halfway between a Combo box and a List box and is described below. Using Drop-Down List Boxes To create a Drop-down list box, choose a Combo box from the control toolbar and place it in your dialog. Double-click the left mouse button on the Combo box to open the Properties box. On the Styles Tab, choose Drop List as the control type. A Drop-down list box has a drop-down arrow to display the list. Like a Combo box, only one selection can be made at a time in the list, but like a List Box, the selected value cannot be edited. A Drop-down list box serves the same function as a List box except for the disadvantage that the user can choose only a single selection, and the advantage that it takes up less dialog screen space. A Drop-down list box has the same control indexes as a Combo box with the addition of another INTEGER index to set or return the list number of the item selected in the list. For example:
INTEGER num ! Returns index of the selected item. retlog = DLGGET (dlg, IDC_DROPDOWN1, num, DLG_STATE)
Using Dialogs
Page 42 of 53
index DLG_RANGEMIN and DLG_RANGEMAX with DLGSET or DLGSETINT. The default values are 1 and 100. For example:
LOGICAL retlog retlog = DLGSET (dlg, IDC_SCROLLBAR1, 212, DLG_RANGEMAX)
You get the slide position by retrieving the control index DLG_POSITION with DLGGET or DLGGETINT. For example:
INTEGER slide_position retlog = DLGGET (dlg, IDC_SCROLLBAR1, slide_position, DLG_POSITION)
You can also set the increment taken when the user clicks in the blank area above or below the slide in a vertical Scroll bar, or to the left or right of the slide in a horizontal Scroll bar, by setting the control index DLG_BIGSTEP. For example:
retlog = DLGSET (dlg, IDC_SCROLLBAR1, 20, DLG_BIGSTEP)
When the user clicks on the arrow buttons of the Scroll bar, the position is always incremented or decremented by 1. The maximum value (DLG_POSITION) that a scroll bar can report (that is, the maximum scrolling position) depends on the page size (DLG_BIGSTEP). If the scroll bar has a page size greater than one, the maximum scrolling position is less than the maximum range value (DLG_RANGEMAX or DLG_RANGE). You can use the following formula to calculate the maximum scrolling position:
MaxScrollPos = MaxRangeValue - (PageSize - 1)
For example, if a scroll bar has DLG_RANGEMAX = 100 (100 is the default value of DLG_RANGEMAX) and DLG_BIGSTEP = 10 (10 is the default value of DLG_BIGSTEP), then the maximum DLG_POSITION is 91 (100 - (10 - 1)). The maximum DLG_POSITION of scroll bars in the Visual Fortran V5 (and Microsoft Fortran PowerStation V4) Dialog Procedures is the same value as the maximum range value (DLG_RANGE). This allows your application to implement a "proportional" scroll bar. The size of the scroll box (or thumb) is determined by the value of DLG_BIGSTEP and should represent a "page" of data (that is, the amount of data visible in the window). When the user clicks in the "shaft" of the scroll bar, the next (or previous) page is displayed. The top of the thumb (for a vertical scroll bar) or the left edge of the thumb (for a horizontal scroll bar) represents the position of the scroll bar (DLG_POSITION). The size of the thumb represents the amount of data currently visible. There is a
Using Dialogs
Page 43 of 53
minimum thumb size so as not to affect usability. When the scroll bar is at its maximum position, the position will represent the position in the data such that the last "page" of data is visible in the window. When the top (or left edge) of the scroll bar is at the mid-point of the shaft, DLG_POSITION will be the midpoint of the range and the mid-point of the data should be displayed at the top of the window.
Using Pictures
The Picture control is an area of your dialog box in which your application displays a picture. The user cannot change it, since it is an output-only window. It does not respond to user input and therefore does not support any callbacks. The picture displayed can be set using the Properties dialog box in the Resource Editor. The options that can be fully defined using the Resource Editor include a:
l l l l
When using the Resource editor, you need to double-click on the border of the picture to display the Picture Properties box. If the Properties box displayed has a title of Dialog Properties instead of Picture Properties, dismiss the Dialog Properties box and try to select the border before you perform the double-click. The Picture Properties box allows you to select the Type of picture as well as specify the Image for certain picture types. For example, to add an existing BMP (Bitmap) file to a dialog: 1. After you open your project workspace, from the Insert menu, click Resource. 2. Select the Resource type (such as Icon or Bitmap) and then select whether it is a new resource, an Import (existing) resource, or Custom resource. In our example, we will specify an existing Bitmap file we have previously copied into our project directory (click Import). 3. After we specify the resource, its name appears in the ResourceView under the appropriate resource type (such as bitmap). 4. Add a picture to the dialog box, by dragging the picture icon from the Control toolbar to the appropriate place in the dialog box. You can resize the picture as needed. The default picture type is frame. 5. Carefully double-click on the border of the picture area to display the Picture Properties box. If the Dialog Properties box appears instead of Picture Properties, dismiss the Dialog Properties box and try again. 6. In the Picture Properties box, select the Type of picture. If you select a
Using Dialogs
Page 44 of 53
type of Bitmap or Icon, for example, you can select the Image from the list of available project resources. 7. You can move the picture as needed. 8. Dismiss the Picture Properties box.
Using Sliders
The Slider Control is a window that contains a slider and optional tick marks. Your application sets or gets the range of the Slider control's values, using
Using Dialogs
DLG_RANGEMIN
Page 45 of 53
and DLG_RANGEMAX, and the current value, using DLG_POSITION. Your application can also set:
l
The number of logical positions the slider moves in response to keyboard input from the arrow keys using DLG_SMALLSTEP. The number of logical positions the slider moves in response to keyboard input, such as the PAGE UP or PAGE DOWN keys, or mouse input, such as clicks in the slider's channel, using DLG_BIGSTEP. The interval frequency for tick marks on the slider using DLG_TICKFREQ.
The Slider Control calls the DLG_CHANGE callback whenever the user changes the current value of the control. The Slider control is named the "Trackbar" control in Windows programming documentation.
You define each of the Tab dialogs using the resource editor as you do for the dialog box that contains the Tab control. In the Dialog Properties, you must make the following style settings for each Tab dialog: 1. Set the "Style" to "Child" 2. Set "Border" to "None"
Page 46 of 53
Before displaying the dialog box that contains the Tab control (using DLGMODAL or DLGMODELESS): 1. Call DLGSETSUB to define a DLG_INIT callback for the dialog box 2. Call DLGINIT for each Tab dialog In the DLG_INIT callback of the dialog box that contains the Tab control, if the callbacktype is DLG_INIT, call DLGMODELESS for each of the Tab dialog boxes. Specify SW_HIDE as the second parameter, and the window handle of the Tab control as the third parameter. After calling DLGMODELESS, call DLGSET with the DLG_STATE index to set the initial Tab. For example:
! When the Main dialog box is first displayed, call DlgModeless to ! display the Tab dialog boxes. Note the use of SW_HIDE. The ! Dialog Functions will "show" the proper Tab dialog box. if (callbacktype == dlg_init) then hwnd = GetDlgItem(dlg % hwnd, IDC_TAB) lret = DlgModeless(gdlg_tab1, SW_HIDE, hwnd) lret = DlgModeless(gdlg_tab2, SW_HIDE, hwnd) lret = DlgModeless(gdlg_tab3, SW_HIDE, hwnd) ! Note that we must set the default Tab after the calls to ! DlgModeless. Otherwise, no Tab dialog box will be displayed ! initially. lret = DlgSet(dlg, IDC_TAB, 1, dlg_state)
Call DLGUNINIT for each Tab dialog when you are done with it. For a complete example of using a Tab control, see the Visual Fortran Sample ShowFont in the ...\DF98\SAMPLES\DIALOG folder.
The only argument for DLGEXIT is the dialog derived type. The dialog box is
Using Dialogs
Page 47 of 53
exited after DLGEXIT returns control back to the dialog manager, not immediately after calling DLGEXIT. That is, if there are other statements following DLGEXIT within the callback routine that contains it, those statements are executed and the callback routine returns before the dialog box is exited. If you want DLGMODAL to return with a value other than the control name of the control that caused the exit, (or -1 if DLGMODAL fails to open the dialog box), you can specify your own return value with the subroutine DLGSETRETURN. For example:
TYPE (DIALOG) dlg INTEGER altreturn ... altreturn = 485 CALL DLGSETRETURN (dlg, altreturn) CALL DLGEXIT(dlg)
To avoid confusion with the default failure condition, use return values other than -1. It is not possible to return a value when a modeless dialog box exits. However, you can call DLGSETSUB to set the DLG_INIT callback routine to have a procedure called immediately before the dialog box is destroyed. If you want the user to be able to close the dialog from the system menu or by pressing the ESC key, you need a control that has the ID of IDCANCEL. When a system escape or close is performed, it simulates pressing the dialog button with the ID IDCANCEL. If no control in the dialog has the ID IDCANCEL, then the close command will be ignored (and the dialog can not be closed in this way). If you want to enable system close or ESC to close a dialog, but do not want a cancel button, you can add a button with the ID IDCANCEL to your dialog and then remove the visible property in the button's Properties box. Pressing ESC will then activate the default click callback of the cancel button and close the dialog.
Using Dialogs
Page 48 of 53
Fortran Module Wizard to generate a module that allows you to use the methods, properties, and events of the ActiveX control from Fortran. 3. Adding Code to Your Application discusses adding code to your application to manipulate the ActiveX control and respond to control events. 4. Target System Requirements describes how to ensure that the ActiveX control and the dialog procedure ActiveX container DLL are present and registered on the target system.
Using Dialogs
Page 49 of 53
Wizard screen. In the Type Library screen, choose the ActiveX control that you want to use. Typically, this is a file with a .OCX extension. See the documentation on the ActiveX control to determine its filename. You may select individual components of the ActiveX control to process using the Module Wizard, but typically you can let the wizard default to processing all of the components. After entering the information and pressing the Generate button, the Fortran Module Wizard asks you for the name of the source file to be generated. It then asks COM to open the type library and generates a file containing Fortran modules. To add the newly created file to your project, in the Project menu, click Add To Project -> Files. For more information:
l
About using the Fortran Module Wizard, see Using COM and Automation Objects.
You do not need to specify the index DLG_IDISPATCH because it is the default integer index for an ActiveX control. Note however, that the control's IDispatch pointer is not available until after the control has been created and is only valid until the dialog box is closed. The control is created during the call to DLGMODAL or DLGMODELESS. If you call DLGGET to retrieve the IDispatch pointer before calling DLGMODAL or DLGMODELESS, the value returned will be 0.
Using Dialogs
Page 50 of 53
Do not call COMRELEASEOBJECT with the iDispatch pointer returned by DLGGET. The dialog procedures use a reference counting optimization since the lifetime of the control is guaranteed to be less than the lifetime of the dialog box. If you want to use a method or property of a control before the dialog box is displayed to your application's user, you can use a DLG_INIT callback. Call DLGSETSUB using the dialog box name and the DLG_INIT index to define the callback. For example:
retlog = DlgSetSub( dlg, IDD_DIALOG, DlgSub, DLG_INIT )
The DLG_INIT callback is called after the dialog box is created but before it is displayed (with callbacktype=DLG_INIT) and immediately before the dialog box is destroyed (with callbacktype=DLG_DESTROY). The DLG_INIT callback is the soonest that the control's IDispatch pointer is available. The DLG_DESTROY callback is the latest that this pointer is valid. After the DLG_DESTROY callback, the ActiveX control is destroyed. The following example shows using a DLG_INIT callback to reset the state of a control property before it is destroyed:
SUBROUTINE mmplayerSub( dlg, id, callbacktype ) !DEC$ ATTRIBUTES DEFAULT :: mmplayerSub use dflogm use dfcom use dfauto implicit none type (dialog) dlg integer id, callbacktype include 'resource.fd' integer obj, iret logical lret if (callbacktype == dlg_init) then lret = DlgGet(dlg, IDC_ACTIVEMOVIECONTROL1, obj) ! Add any method or property calls here before the ! dialog box is displayed else if (callbacktype == dlg_destroy) then ! Reset the filename to "" to "close" the current file lret = DlgGet(dlg, IDC_ACTIVEMOVIECONTROL1, obj) iret = AUTOSETPROPERTY(obj, "FileName", "") endif END SUBROUTINE mmplayerSub
The module generated by the Fortran Module Wizard for an ActiveX control contains a number of sections:
l
! CLSIDs
Parameters of derived type GUID which identify the ActiveX control class. Your application typically doesn't need to use this parameter.
l
! IIDs
Using Dialogs
Page 51 of 53
of the ActiveX control. Your application can use these values in calls to DLGSETCTRLEVENTHANDLER (see below).
l
! Enums
Parameters of type integer that identify constants used in the ActiveX control's interfaces.
l
! Interfaces
Interfaces for the source (event) interfaces that are defined by the ActiveX control. There may be 0, 1, or more source interfaces implemented by the control. A control does not have to support events.
l
! Module Procedures
Wrapper routines that make it easy to call the control's methods and get or retrieve the control's properties. See Calling the Routines Generated by the Module Wizard for more information on using the method and property interfaces generated by the Fortran Module Wizard. In addition to methods and properties, ActiveX controls also define events to notify your application that something has happened to the control. The dialog procedures provide a routine, DLGSETCTRLEVENTHANDLER, that allows you to define a routine to be executed when an event occurs. The DLGSETCTRLEVENTHANDLER function has the following interface:
integer DlgSetCtrlEventHandler( dlg, controlid, handler, dispid, iid )
(Input) Derived type DIALOG. Contains dialog box parameters. (Input) Integer. Specifies the identifier of a control within the dialog box (from the .FD file). (Input) EXTERNAL. Name of the routine to be called when the event occurs. (Input) Integer. Specifies the member id of the method in the event interface that identifies the event (Input, Optional) Derived type(GUID). Specifies the Interface identifier of the source (event) interface. If not supplied, the default source interface of the ActiveX control is used.
handler
dispid
iid
Using Dialogs
Page 52 of 53
IDC_ACTIVEMOVIECONTROL1 identifies an ActiveMovie control in the dialog box. ActiveMovie_ReadyStateChange is the name of the event handling routine. -609 is the member id of the ActiveMovie's control ReadyStateChange
event. You can get this number from: The module that the Fortran Module Wizard generated. There is a "MEMBERID = nn" comment generated for each method in a source interface (see the example below). The documentation of the ActiveX control. A tool that allows you to examine the type information of the ActiveX control, for example, the OLE-COM Object Viewer in the Visual Fortran folder. IID_DActiveMovieEvents2 is the identifier of the source (event) interface.
The interface generated by the Fortran Module Wizard for the ReadyStateChange event follows:
INTERFACE !Reports that the ReadyState property of the ActiveMovie Control has changed ! MEMBERID = -609 SUBROUTINE DActiveMovieEvents2_ReadyStateChange($OBJECT, ReadyState) INTEGER(4), INTENT(IN) :: $OBJECT ! Object Pointer !DEC$ ATTRIBUTES VALUE :: $OBJECT INTEGER(4) :: ReadyState !DEC$ ATTRIBUTES VALUE :: ReadyState !DEC$ ATTRIBUTES STDCALL :: DActiveMovieEvents2_ReadyStateChange END SUBROUTINE DActiveMovieEvents2_ReadyStateChange END INTERFACE
The handler that you define in your application must have the same interface. Otherwise, your application will likely crash in unexpected ways because of the application's stack getting corrupted. Note that an object is always the first parameter in an event handler. This object value is a pointer to the control's source (event) interface, not the IDispatch pointer of the control. You can use DLGGET as described above to retrieve the control's IDispatch pointer.
Using Dialogs
Page 53 of 53
To register a DLL, use REGSVR32.EXE, located in the Windows system directory. REGSVR32 takes a single argument: the path of the DLL. You can alternatively use VFRUNxxx to install the complete set of Visual Fortran run-time files. You can download Visual Fortran redistributable files (VFRUNxxx) from the download/update area of Compaq Fortran Web site: http://www.compaq.com/fortran
Page 1 of 24
l l l
The origin (point 0, 0) is the upper-left corner of the screen or the client area (defined user area) of the child window being written to. The x-axis and y-axis start at the origin. You can change the origin in some coordinate systems. The horizontal direction is represented by the x-axis, increasing to the right. The vertical direction is represented by the y-axis, increasing down. Some graphics adapters offer a color palette that can be changed. Some graphics adapters (VGA and SVGA) allow you to change the color that a color index refers to by providing a color value that describes a new color. The color value indicates the mix of red, green, and blue in a screen color. A color value is always an INTEGER(4) number.
Working with Graphics Modes Adding Color Understanding Coordinate Systems Advanced Graphics Using OpenGL
Checking the Current Graphics Mode Setting the Graphics Mode Writing a Graphics Program
Page 2 of 24
TYPE windowconfig INTEGER(2) numxpixels INTEGER(2) numypixels INTEGER(2) numtextcols INTEGER(2) numtextrows INTEGER(2) numcolors INTEGER(4) fontsize CHARACTER(80) title INTEGER(2) bitsperpixel END TYPE windowconfig
! ! ! ! ! ! ! !
Number of pixels on x-axis Number of pixels on y-axis Number of text columns available Number of text rows available Number of color indexes Size of default font window title Number of bits per pixel
By default, a QuickWin child window is a scrollable text window 640x480 pixels, has 30 lines and 80 columns, and a font size of 8x16. Also by default, a Standard Graphics window is Full Screen. You can change the values of window properties at any time with SETWINDOWCONFIG, and retrieve the current values at any time with GETWINDOWCONFIG.
If you use SETWINDOWCONFIG, you should specify a value for each field ( 1
Page 3 of 24
or your own number for numeric fields, and a C string for the title). Calling SETWINDOWCONFIG with only some of the fields specified can result in useless values for the other fields.
graphicsmode in section Activating a Graphics Mode drawlines in section Drawing Lines on the Screen sinewave in section Drawing a Sine Curve drawshapes in section Adding Shapes
The SINE program's output appears in the following figure. The SINE routines are in the Visual Fortran Sample TUTORIAL folder. The project is built as a Fortran Standard Graphics application. Sine Program Output
Page 4 of 24
Pixel coordinates start at zero, so, for example, a screen with a resolution of 640 horizontal pixels has a maximum x-coordinate of 639. Thus, maxx (the highest available x-pixel coordinate) must be 1 less than the total number of pixels. The same applies to maxy. To remain independent of the video mode set by graphicsmode, two short functions convert an arbitrary screen size of 1000x1000 pixels to whatever video mode is in effect. From now on, the program assumes it has 1000 pixels in each direction. To draw the points on the screen, newx and newy map each point to their physical (pixel) coordinates:
Page 5 of 24
! NEWX - This function finds new x-coordinates. INTEGER(2) FUNCTION newx( xcoord ) INTEGER(2) xcoord, maxx, maxy REAL(4) tempx COMMON maxx, maxy tempx = maxx / 1000.0 tempx = xcoord * tempx + 0.5 newx = tempx END FUNCTION ! NEWY - This function finds new y-coordinates. ! INTEGER(2) FUNCTION newy( ycoord ) INTEGER(2) ycoord, maxx, maxy REAL(4) tempy COMMON maxx, maxy tempy = maxy / 1000.0 tempy = ycoord * tempy + 0.5 newy = tempy END FUNCTION
You can set up a similar independent coordinate system with window coordinates, described in Understanding Coordinate Systems.
SUBROUTINE drawlines( ) USE DFLIB EXTERNAL INTEGER(2) TYPE (xycoord) COMMON ! ! Draw the box. status = RECTANGLE( $GBORDER, INT2(0), INT2(0), maxx, maxy ) CALL SETVIEWORG( INT2(0), newy( INT2( 500 ) ), xy ) ! This sets the new origin to 0 for x and 500 for y. See comment after subroutine. Draw the lines. CALL MOVETO( INT2(0), INT2(0), xy ) status = LINETO( newx( INT2( 1000 )), INT2(0)) CALL SETLINESTYLE( INT2( #AA3C )) CALL MOVETO( INT2(0), newy( INT2( -250 )), xy ) status = LINETO(newx( INT2( 1000 )),newy( INT2( -250 ))) CALL SETLINESTYLE( INT2( #8888 )) newx, newy status, newx, newy, maxx, maxy xy maxx, maxy
! !
Page 6 of 24
The first argument to RECTANGLE is the fill flag, which can be either $GBORDER or $GFILLINTERIOR. Choose $GBORDER if you want a rectangle of four lines (a border only, in the current line style), or $GFILLINTERIOR if you want a solid rectangle (filled in with the current color and fill pattern). Choosing the color and fill pattern is discussed in Adding Color and Adding Shapes. The second and third RECTANGLE arguments are the x- and y-coordinates of the upper-left corner of the rectangle. The fourth and fifth arguments are the coordinates for the lower-right corner. Because the coordinates for the two corners are ( 0, 0 ) and ( maxx, maxy ), the call to RECTANGLE frames the entire screen. The program calls SETVIEWORG to change the location of the viewport origin. By resetting the origin to (0, 500) in a 1000x1000 viewport, you effectively make the viewport run from (0, -500) at the top left of the screen to (1000, 500) at the bottom right of the screen:
CALL SETVIEWORG( INT2(0), newy( INT2( 500 ) ), xy )
Changing the coordinates illustrates the ability to alter the viewport coordinates to whatever dimensions you prefer. (Viewports and the SETVIEWORG routine are explained in more detail in Understanding Coordinate Systems.) The call to SETLINESTYLE changes the line style from a solid line to a dashed line. A series of 16 bits tells the routine which pattern to follow. A "1" indicates a solid pixel and "0" an empty pixel. Therefore, 1111 1111 1111 1111 represents a solid line. A dashed line might look like 1111 1111 0000 0000 (long dashes) or 1111 0000 1111 0000 (short dashes). You can choose any combination of ones and zeros. Any INTEGER(2) number in any base is an acceptable input, but binary and hexadecimal numbers are easier to envision as line-style patterns. In the example, the hexadecimal constant #AA3C equals the binary value 1010 1010 0011 1100. You can use the decimal value 43580 just as effectively. When drawing lines, first set an appropriate line style. Then, move to where you want the line to begin and call LINETO, passing to it the point where you want the line to end. The drawlines subroutine uses the following code:
CALL SETLINESTYLE(INT2( #AA3C ) ) CALL MOVETO( INT2(0), newy( INT2( -250 ) ), xy ) dummy = LINETO( newx( INT2( 1000 )), newy( INT2( -250 )))
MOVETO positions an imaginary pixel cursor at a point on the screen (nothing appears on the screen), and LINETO draws a line. When the program called SETVIEWORG, it changed the viewport origin, and the initial y-axis range of 0
Page 7 of 24
to 1000 now corresponds to a range of 500 to +500. Therefore, the negative value 250 is used as the y-coordinate of LINETO to draw a horizontal line across the center of the top half of the screen, and the value of 250 is used as the y-coordinate to draw a horizontal line across the center of the bottom half of the screen.
Calculate each position and display it on the screen. color = #0000FF ! red
SETPIXELRGB takes the two location parameters, locx and locy, and sets the pixel at that position with the specified color value (red).
Adding Shapes
After drawing the sine curve, SINE calls drawshapes to put two rectangles and two ellipses on the screen. The fill flag alternates between $GBORDER and $GFILLINTERIOR:
! DRAWSHAPES - Draws two boxes and two ellipses. ! SUBROUTINE drawshapes( ) USE DFLIB EXTERNAL INTEGER(2) ! ! ! newx, newy dummy, newx, newy
Page 8 of 24
The call to SETLINESTYLE resets the line pattern to a solid line. Omitting this routine causes the first rectangle to appear with a dashed border, because the drawlines subroutine called earlier changed the line style to a dashed line. ELLIPSE draws an ellipse using parameters similar to those for RECTANGLE. It, too, requires a fill flag and two corners of a bounding rectangle. The following figure shows how an ellipse uses a bounding rectangle: Bounding Rectangle
The $GFILLINTERIOR constant fills the shape with the current fill pattern. To create a pattern, pass the address of an 8-byte array to SETFILLMASK. In drawshapes, the diagmask array is initialized with the pattern shown in the following table: Fill Patterns
Bit pattern Bit No.7 6 5 x o o x x o o x x x o x o x o o o x x o o Value in diagmask 4 x o o x x o x 3 o x o o x x o 2 o o x o o x x 1 x o o x o o x 0 x x o o x o o diagmask(1) diagmask(2) diagmask(3) diagmask(4) diagmask(5) diagmask(6) diagmask(7) = = = = = = = #93 #C9 #64 #B2 #59 #2C #96
Page 9 of 24
Adding Color
The Visual Fortran QuickWin Library supports color graphics. The number of total available colors depends on the current video driver and video adapter you are using. The number of available colors you use depends on the graphics functions you choose. The different color modes and color functions are discussed and demonstrated in the following sections:
l l l
Color Mixing
If you have a VGA machine, you are restricted to displaying at most 256 colors at a time. These 256 colors are held in a palette. You can choose the palette colors from a range of 262,144 colors (256K), but only 256 at a time. Some display adapters (most SVGAs) are capable of displaying all of the 256K colors and some (true color display adapters) are capable of displaying 256 * 256 * 256 = 16.7 million colors. If you use a palette, you are restricted to the colors available in the palette. In order to access all colors available on your system, you need to specify an explicit Red-Green-Blue (RGB) value, not a palette index. When you select a color index, you specify one of the colors in the system's predefined palette. SETCOLOR, SETBKCOLOR, and SETTEXTCOLOR set the current color, background color, and text color to a palette index. SETCOLORRGB, SETBKCOLORRGB, and SETTEXTCOLORRGB set the colors to a color value chosen from the entire available range. When you select a color value, you specify a level of intensity with a range of 0 - 255 for each of the red, green, and blue color values. The long integer that defines a color value consists of 3 bytes (24 bits) as follows:
MSB LSB BBBBBBBB GGGGGGGG RRRRRRRR
where R, G, and B represent the bit values for red, green, and blue intensities. To mix a light red (pink), turn red all the way up and mix in some green and blue:
10000000 10000000 11111111
In hexadecimal notation, this number equals #8080FF. To set the current color
Drawing Graphics Elements to this value, you can use the function:
i = SETCOLORRGB (#8080FF)
Page 10 of 24
You can also pass decimal values to this function. Keep in mind that 1 (binary 00000001, hex 01) represents a low color intensity and that 255 (binary 11111111, hex FF) equals full color intensity. To create pure yellow (100percent red plus 100-percent green) use this line:
i = SETCOLORRGB( #00FFFF )
RGB values for example colors are in the following table. RGB Color Values Color Black Dull Red Dull Green Dull Yellow Dull Blue Dull Magenta Dull Turquoise Dark Gray RGB Value #000000 #000080 #008000 #008080 #800000 #800080 #808000 #808080 Color Bright White Bright Red Bright Green Bright Yellow Bright Blue Bright Magenta Bright Turquoise Light Gray RGB Value #FFFFFF #0000FF #00FF00 #00FFFF #FF0000 #FF00FF #FFFF00 #C0C0C0
If you have a 64K-color machine and you set an RGB color value that is not equal to one of the 64K preset RGB color values, the system approximates the requested RGB color to the closest available RGB value. The same thing happens on a VGA machine when you set an RGB color that is not in the palette. (You can remap your VGA color palette to different RGB values; see VGA Color Palette.)
Page 11 of 24
However, although your graphics are drawn with an approximated color, if you retrieve the color with GETCOLORRGB, GETBKCOLORRGB, or GETTEXTCOLORRGB, the color you specified is returned, not the actual color used. This is because the SETCOLORRGB functions do not execute any graphics, they simply set the color and the approximation is made when the drawing is made (by ELLIPSE or ARC, for example). GETPIXELRGB and GETPIXELSRGB do return the approximated color actually used, because SETPIXELRGB and SETPIXELSRGB actually set a pixel to a color on the screen and the approximation, if any, is made at the time they are called.
REMAPALLPALETTERGB remaps one or more color indexes simultaneously. Its argument is an array of RGB color values that are mapped into the palette. The first color number in the array becomes the new color associated with color index 0, the second with color index 1, and so on. At most 236 indexes can be mapped, because 20 indexes are reserved for system use. If you request an RGB color that is not in the palette, the color selected from the palette is the closest approximation to the RGB color requested. If the RGB color was previously placed in the palette with REMAPPALETTERGB or REMAPALLPALETTERGB, then that exact RGB color is available. Remapping the palette has no effect on 64K-color machines, SVGA, or true-color machines, unless you limit yourself to a palette by using color index functions
Page 12 of 24
such as SETCOLOR. On a VGA machine, if you remap all the colors in your palette and display that palette in graphics, you cannot then remap and simultaneously display a second palette. For instance, in VGA 256-color mode, if you remap all 256 palette colors and display graphics in one child window, then open another child window, remap the palette and display graphics in the second child window, you are attempting to display more than 256 colors at one time. The machine cannot do this, so whichever child window has the focus will appear correct, while the one without the focus will change color. Note: Machines that support more than 256 colors will not be able to do animation by remapping the palette. Windows operating systems create a logical palette that maps to the video hardware palette. On video hardware that supports a palette of 256 colors or less, remapping the palette maps over the current palette and redraws the screen in the new colors. On large hardware palettes that support more than 256 colors, remapping is done into the unused portion of the palette. It does not map over the current colors nor redraw the screen. So, on machines with large palettes (more than 256 colors), the technique of changing the screen through remapping, called palette animation, cannot be used. See the Platform SDK Manual for more information.
Symbolic constants (names) for the default color numbers are supplied in the graphics modules. The names are self-descriptive; for example, the color numbers for black, yellow, and red are represented by the symbolic constants $BLACK, $YELLOW, and $RED.
Page 13 of 24
Text Coordinates
The text modes use a coordinate system that divides the screen into rows and columns as shown in the following figure: Text Screen Coordinates
Numbering starts at 1. An 80-column screen contains columns 1-80. The row is always listed before the column.
If the screen displays 25 rows and 80 columns (as shown in the above Figure), the rows are numbered 1-25 and the columns are numbered 1-80. The textpositioning routines, such as SETTEXTPOSITION and SCROLLTEXTWINDOW, use row and column coordinates.
Graphics Coordinates
Three coordinate systems describe the location of pixels on the screen:
l l l
Page 14 of 24
In all three coordinate systems, the x-coordinate is listed before the ycoordinate.
Physical Coordinates Physical coordinates are integers that refer to pixels in a window's client area. By default, numbering starts at 0, not 1. If there are 640 pixels, they are numbered 0-639. Suppose your program calls SETWINDOWCONFIG to set up a client area containing 640 horizontal pixels and 480 vertical pixels. Each individual pixel is referred to by its location relative to the x-axis and y-axis, as shown in the following figure:
Physical Coordinates
The upper-left corner is the origin. The x- and y-coordinates for the origin are always (0, 0). Physical coordinates refer to each pixel directly and are therefore integers (that is, the window's client area cannot display a fractional pixel). If you use variables to refer to pixel locations, declare them as integers or use typeconversion routines when passing them to graphics functions. For example:
ISTATUS = LINETO( INT2(REAL_x), INT2(REAL_y))
If a program uses the default dimension of a window, the viewport (drawing area) is equal to 640x480. SETVIEWORG changes the location of the viewport's origin. You pass it two integers, which represent the x and y physical screen coordinates for the new origin. You also pass it an xycoord type that the routine fills with the physical coordinates of the previous origin. For example, the following line moves the viewport origin to the physical screen location (50, 100):
TYPE (xycoord) origin CALL SETVIEWORG(INT2(50), INT2(100), origin)
Page 15 of 24
The number of pixels hasn't changed, but the coordinates used to refer to the points have changed. The x-axis now ranges from -50 to +589 instead of 0 to 639. The y-axis now covers the values -100 to +379. All graphics routines that use viewport coordinates are affected by the new origin, including MOVETO, LINETO, RECTANGLE, ELLIPSE, POLYGON, ARC, and PIE. For example, if you call RECTANGLE after relocating the viewport origin and pass it the values (0, 0) and (40, 40), the upper-left corner of the rectangle would appear 50 pixels from the left edge of the screen and 100 pixels from the top. It would not appear in the upper-left corner of the screen. SETCLIPRGN creates an invisible rectangular area on the screen called a clipping region. You can draw inside the clipping region, but attempts to draw outside the region fail (nothing appears outside the clipping region). The default clipping region occupies the entire screen. The QuickWin Library ignores any attempts to draw outside the screen. You can change the clipping region by calling SETCLIPRGN. For example, suppose you entered a screen resolution of 320x200 pixels. If you draw a diagonal line from (0, 0) to (319, 199), the upper-left to the lower-right corner, the screen looks like the following figure:
Line Drawn on a Full Screen
With the clipping region in effect, the same LINETO command would put the line shown in the following figure on the screen:
Line Drawn Within a Clipping Region
Page 16 of 24
The dashed lines indicate the outer bounds of the clipping region and do not actually print on the screen. Viewport Coordinates The viewport is the area of the screen displayed, which may be only a portion of the window's client area. Viewport coordinates represent the pixels within the current viewport. SETVIEWPORT establishes a new viewport within the boundaries of the physical client area. A standard viewport has two distinguishing features:
l l
The origin of a viewport is in the upper-left corner. The default clipping region matches the outer boundaries of the viewport.
SETVIEWPORT has the same effect as SETVIEWORG and SETCLIPRGN combined. It specifies a limited area of the screen in the same manner as SETCLIPRGN, then sets the viewport origin to the upper-left corner of the area. Window Coordinates Functions that refer to coordinates on the client-area screen and within the viewport require integer values. However, many applications need floating-point valuesfor frequency, viscosity, mass, and so on. SETWINDOW lets you scale the screen to almost any size. In addition, window-related functions accept double-precision values. Window coordinates use the current viewport as their boundaries. A window overlays the current viewport. Graphics drawn at window coordinates beyond the boundaries of the windowthe same as being outside the viewportare clipped. For example, to graph 12 months of average temperatures on the planet Venus that range from 50 to +450, add the following line to your program:
status = SETWINDOW(.TRUE., 1.0D0, -50.0D0, 12.0D0, 450.0D0)
The first argument is the invert flag, which puts the lowest y value in the lowerleft corner. The minimum and maximum x- and y-coordinates follow; the decimal point marks them as floating-point values. The new organization of the
Page 17 of 24
January and December plot on the left and right edges of the screen. In an application like this, numbering the x-axis from 0.0 to 13.0 provides some padding space on the sides and would improve appearance. If you next plot a point with SETPIXEL_W or draw a line with LINETO_W, the values are automatically scaled to the established window. To use window coordinates with floating-point values: 1. Set a graphics mode with SETWINDOWCONFIG. 2. Use SETVIEWPORT to create a viewport area. This step is not necessary if you plan to use the entire screen. 3. Create a real-coordinate window with SETWINDOW, passing a LOGICAL invert flag and four DOUBLE PRECISION x- and y-coordinates for the minimum and maximum values. 4. Draw graphics shapes with RECTANGLE_W and similar routines. Do not confuse RECTANGLE (the viewport routine) with RECTANGLE_W (the window routine for drawing rectangles). All window function names end with an underscore and the letter W ( _W). Real-coordinate graphics give you flexibility and device independence. For example, you can fit an axis into a small range (such as 151.25 to 151.45) or into a large range (50000.0 to +80000.0), depending on the type of data you graph. In addition, by changing the window coordinates, you can create the effects of zooming in or panning across a figure. The window coordinates also make your drawings independent of the computer's hardware. Output to the viewport is independent of the actual screen resolution.
Page 18 of 24
Set the screen to the best resolution and maximum number of available colors. myscreen.numxpixels = -1 myscreen.numypixels = -1 myscreen.numtextcols = -1 myscreen.numtextrows = -1 myscreen.numcolors = -1 myscreen.fontsize = -1 myscreen.title = " "C statusmode = SETWINDOWCONFIG(myscreen) IF(.NOT. statusmode) statusmode = SETWINDOWCONFIG(myscreen) statusmode = GETWINDOWCONFIG( myscreen ) CALL threegraphs( ) END . . .
The main body of the program is very short. It sets the window to the best resolution of the graphics driver (by setting the first four fields to -1) and the maximum number of colors (by setting numcolors to 1). The program then calls the threegraphs subroutine that draws three graphs. The program output is shown in the following figure:
REALG Program Output
The gridshape subroutine, which draws the graphs, uses the same data in each case. However, the program uses three different coordinate windows. The two viewports in the top half are the same size in physical coordinates, but have different window sizes. Each window uses different maximum and
Page 19 of 24
minimum values. In all three cases, the graph area is two units wide. The window in the upper-left corner has a range in the x-axis of four units (4 units wide); the window in the upper-right corner has a range in the x-axis of six units, which makes the graph on the right appear smaller. In two of the three graphs, one of the lines goes off the edge, outside the clipping region. The lines do not intrude into the other viewports, because defining a viewport creates a clipping region. Finally, the graph on the bottom inverts the data with respect to the two graphs above it. The next section describes and discusses the subroutine invoked by REALG.F90: Drawing the Graphs The main program calls threegraphs, which prints the three graphs:
SUBROUTINE threegraphs() USE DFLIB INTEGER(2) status, halfx, halfy INTEGER(2) xwidth, yheight, cols, rows TYPE (windowconfig) myscreen COMMON myscreen CALL CLEARSCREEN( $GCLEARSCREEN ) xwidth = myscreen.numxpixels yheight = myscreen.numypixels cols = myscreen.numtextcols rows = myscreen.numtextrows halfx = xwidth / 2 halfy = (yheight / rows) * ( rows / 2 ) ! ! ! First window CALL SETVIEWPORT( INT2(0), INT2(0), halfx - 1, halfy - 1 ) CALL SETTEXTWINDOW( INT2(1), INT2(1), rows / 2, cols / 2 ) status = SETWINDOW( .FALSE., -2.0_8, -2.0_8, 2.0_8, 2.0_8) The 2.0_8 notation makes these constants REAL(8) CALL gridshape( rows / 2 ) status = RECTANGLE( $GBORDER,INT2(0),INT2(0),halfx-1,halfy-1) ! ! ! Second window CALL SETVIEWPORT( halfx, INT2(0), xwidth - 1, halfy - 1 ) CALL SETTEXTWINDOW( INT2(1), (cols/2) + 1, rows/2, cols) status = SETWINDOW( .FALSE., -3.0D0, -3.0D0, 3.0D0, 3.0D0) The 3.0D0 notation makes these constants REAL(8) CALL gridshape( rows / 2 ) status = RECTANGLE_W( $GBORDER, -3.0_8,-3.0_8,3.0_8, 3.0_8) ! ! ! Third window CALL SETVIEWPORT( 0, halfy, xwidth - 1, yheight - 1 ) CALL SETTEXTWINDOW( (rows / 2 ) + 1, 1_2, rows, cols )
Page 20 of 24
Although the screen is initially clear, threegraphs makes sure by calling the CLEARSCREEN routine to clear the window:
CALL CLEARSCREEN( $GCLEARSCREEN )
The $GCLEARSCREEN constant clears the entire window. Other options include $GVIEWPORT and $GWINDOW, which clear the current viewport and the current text window, respectively. After assigning values to some variables, threegraphs creates the first window:
CALL SETVIEWPORT( INT2(0), INT2(0), halfx - 1, halfy - 1) CALL SETTEXTWINDOW( INT2(1), INT2(1), rows / 2, cols / 2) status = SETWINDOW( .FALSE., -2.0_8, -2.0_8, 2.0_8, 2.0_8)
The first instruction defines a viewport that covers the upper-left quarter of the screen. The next instruction defines a text window within the boundaries of that border. Finally, the third instruction creates a window with both x and y values ranging from 2.0 to 2.0. The .FALSE. constant causes the y-axis to increase from top to bottom, which is the default. The _8 notation identifies the constants as REAL(8). Next, the function gridshape inserts the grid and plots the data, and a border is added to the window:
CALL gridshape( rows / 2 ) status = RECTANGLE( $GBORDER,INT2(0),INT2(0),halfx-1,halfy-1)
This is the standard RECTANGLE routine, which takes coordinates relative to the viewport, not the window. The gridshape subroutine plots the data on the screen.
! GRIDSHAPE - This subroutine plots data for REALG.F90 ! SUBROUTINE gridshape( numc ) USE DFLIB INTEGER(2) INTEGER(4) CHARACTER(8) REAL(8) TYPE (windowconfig) TYPE (wxycoord) TYPE (rccoord) COMMON ! ! Data for the graph: numc, i, status rgbcolor, oldcolor str bananas(21), x myscreen wxy curpos myscreen
Page 21 of 24
oldcolor = SETCOLORRGB( #0000FF ) ! full red status = RECTANGLE_W( $GBORDER, -1.00_8, -1.00_8, 1.00_8,1.00_8) ! constants made REAL(8) by appending _8 status = RECTANGLE_W( $GBORDER, -1.02_8, -1.02_8, 1.02_8, 1.02_8) ! ! Plot the points. ! x = -0.90 DO i = 1, 19 oldcolor = SETCOLORRGB( #00FF00 ) ! full green CALL MOVETO_W( x, -1.0_8, wxy ) status = LINETO_W( x, 1.0_8 ) CALL MOVETO_W( -1.0_8, x, wxy ) status = LINETO_W( 1.0_8, x ) oldcolor = SETCOLORRGB( #FF0000 ) ! full blue CALL MOVETO_W( x - 0.1_8, bananas( i ), wxy ) status = LINETO_W( x, bananas( i + 1 ) ) x = x + 0.1 END DO CALL MOVETO_W( 0.9_8, bananas( i ), wxy ) status = LINETO_W( 1.0_8, bananas( i + 1 ) ) oldcolor = SETCOLORRGB( #00FFFF ) ! yellow END SUBROUTINE
The routine names that end with _W work in the same way as their viewport equivalents, except that you pass double-precision floating-point values instead of integers. For example, you pass INTEGER(2) to LINETO, but REAL(8) values to LINETO_W. The two other windows are similar to the first. All three call the gridshape function, which draws a grid from location (1.0, 1.0) to (1.0, 1.0). The grid appears in different sizes because the coordinates in the windows vary. The second window ranges from (3.0, 3.0) to (3.0, 3.0), and the third from (3.0, 1.5) to (1.5, 1.5), so the sizes change accordingly. The third window also contains a .TRUE. inversion argument. This causes the yaxis to increase from bottom to top, instead of top to bottom. As a result, this graph appears upside down with respect to the other two.
Page 22 of 24
After calling gridshape, the program frames each window, using a statement such as the following:
status = RECTANGLE_W( $GBORDER, -3.0_8, -1.5_8, 1.5_8, 1.5_8)
The first argument is a fill flag indicating whether to fill the rectangle's interior or just to draw its outline. The remaining arguments are the x and y coordinates for the upper-left corner followed by the x and y coordinates for the lower-right corner. RECTANGLE takes integer arguments that refer to the viewport coordinates. RECTANGLE_W takes four double-precision floating-point values referring to window coordinates. After you create various graphics elements, you can use the font-oriented routines to polish the appearance of titles, headings, comments, or labels. Using Fonts from the Graphics Library describes in more detail how to print text in various fonts with font routines.
Specify the pixel format Specify how the pixels will be rendered on the video device Call OpenGL commands
OpenGL programming is straightforward, but requires a particular initialization and order, like other software tools. References to get you started are:
l l l
l l
The OpenGL Reference Manual, Addison-Wesley, ISBN 0-201-46140-4. The OpenGL Programming Guide, Addison-Wesley, ISBN 0-201-46138-2. OpenGL SuperBible: The Complete Guide to OpenGL Programming on Windows NT and Windows 95, Richard Wright and Michael Sweet, Waite Group Press (Division of Sams Publishing), 1996, ISBN 1-57169-073-5. OpenGL documentation in the Platform SDK title in HTML Help Viewer. The OpenGL description from the Microsoft Visual C++ manuals.
Page 23 of 24
Visual Fortran provides an OpenGL module, DFOPNGL.MOD, invoked with the USE statement line:
USE DFOPNGL
When you use this module, all constants and interfaces that bind Fortran to the OpenGL routines become available. Any link libraries required to link with an OpenGL program are automatically searched if USE DFOPNGL is present in your Fortran program. An OpenGL window can be opened from a console, Windows, or QuickWin application. The OpenGL window uses OpenGL calls exclusively, not normal Graphic Device Interface (GDI) calls. Likewise, OpenGL calls cannot be made within an ordinary Windows window or QuickWin child window, because special initialization is required for OpenGL calls. The Fortran OpenGL identifiers are the same as the C identifiers (such as using a GL_ prefix for constants), except that the gl prefix is changed to fgl for routines. The data types in the OpenGL C binding are translated to Fortran types as shown in the following table: OpenGL/C Type GLbyte GLshort GLint, GLsizei GLfloat, GLclampf GLdouble, GLclampd GLubyte GLboolean GLushort GLuint, GLenum, GLbitfield GLvoid pointers Fortran Data Type INTEGER(1) INTEGER(2) INTEGER(4) REAL(4) REAL(8) INTEGER(1) LOGICAL INTEGER(2) INTEGER(4) not needed INTEGER
If you include (USE) the parameter constant definitions from DFOPNGLT.F90 (such as by USE DFOPNGL), you can use the constants to specify the kind type, such as
Page 24 of 24
instead of INTEGER(4).
Visual Fortran Samples that use OpenGL are available in separate folders in ...\DF98\SAMPLES\ADVANCED\OPENGL. For example, OLYMPIC is a Fortran QuickWin project and CUBE5 is a Fortran Windows project.
Page 1 of 5
Available Typefaces
A font is a set of text characters of a particular size and style. A typeface (or type style) refers to the style of the displayed textArial, for example, or Times New Roman. Type size measures the screen area occupied by individual characters. The term comes from the printer's lexicon, but uses screen pixels as the unit of measure rather than the traditional points. For example, "Courier 12 9" denotes the Courier typeface, with each character occupying a screen area of 12 vertical pixels by 9 horizontal pixels. The word "font", therefore implies both a typeface and a type size. The QuickWin Library's font routines use all Windows operating system installed fonts. The first type of font used is a bitmap (or raster-map) font. Bitmap fonts have each character in a binary data map. Each bit in the map corresponds to a screen pixel. If the bit equals 1, its associated pixel is set to the current screen color. Bit values of 0 appear in the current background color. The second type of font is called a TrueType font. Some screen fonts look
Page 2 of 5
different on a printer, but TrueType fonts print exactly as they appear on the screen. TrueType fonts may be bitmaps or soft fonts (fonts that are downloaded to your printer before printing), depending on the capabilities of your printer. TrueType fonts are scalable and can be sized to any height. It is recommended that you use TrueType fonts in your graphics programs. Each type of font has advantages and disadvantages. Bitmapped characters appear smoother on the screen because of the predetermined pixel mapping. However, they cannot be scaled. You can scale TrueType text to any size, but the characters sometimes do not look quite as solid as the bitmapped characters on the screen. Usually this screen effect is hardly noticeable, and when printed, TrueType fonts are as smooth or smoother than bitmapped fonts. The bitmapped typefaces come in preset sizes measured in pixels. The exact size of any font depends on screen resolution and display type.
Using Fonts
QuickWin's font routines can use all the Windows operating system installed fonts. To use fonts in your program, you must: 1. Initialize the fonts. 2. Select a current font from the initialized fonts. 3. Display font text with OUTGTEXT.
Initializing Fonts
A program that uses fonts must first organize the fonts into a list in memory, a process called initializing. The list gives the computer information about the available fonts. Initialize the fonts by calling the INITIALIZEFONTS routine:
USE DFLIB INTEGER(2) numfonts numfonts = INITIALIZEFONTS( )
If the computer successfully initializes one or more fonts, INITIALIZEFONTS returns the number of fonts initialized. If the function fails, it returns a negative error code.
Page 3 of 5
SETFONT(options) The function's argument consists of letter codes that describe the desired font: typeface, character height and width in pixels, fixed or proportional, and attributes such as bold or italic. These options are discussed in detail in the SETFONT entry in the Language Reference. For example:
USE DFLIB INTEGER(2) index, numfonts numfonts = INITIALIZEFONTS ( ) index = SETFONT('t''Cottage''h18w10')
This sets the typeface to Cottage, the character height to 18 pixels and the width to 10 pixels. The following example sets the typeface to Arial, the character height to 14, with proportional spacing and italics (the pi codes):
index = SETFONT('t''Arial''h14pi')
If SETFONT successfully sets the font, it returns the font's index number. If the function fails, it returns a negative integer. Call GRSTATUS to find the source of the problem; its return value indicates why the function failed. If you call SETFONT before initializing fonts, a run-time error occurs. SETFONT updates the font information when it is used to select a font. GETFONTINFO can be used to obtain information about the currently selected font. SETFONT sets the user fields in the fontinfo type (a derived type defined in DFLIB.MOD), and GETFONTINFO returns the user-selected values. The following user fields are contained in fontinfo:
TYPE fontinfo INTEGER(2) type ! 1 = truetype, 0 = bit map INTEGER(2) ascent ! Pixel distance from top to baseline INTEGER(2) pixwidth ! Character width in pixels, 0=prop INTEGER(2) pixheight ! Character height in pixels INTEGER(2) avgwidth ! Average character width in pixels CHARACTER(32) facename ! Font name END TYPE fontinfo
To find the parameters of the current font, call GETFONTINFO. For example:
USE DFLIB TYPE (fontinfo) font INTEGER(2) i, numfonts numfonts = INITIALIZEFONTS() i = SETFONT ( ' t ' 'Arial ' ) i = GETFONTINFO(font) WRITE (*,*) font.avgwidth, font.pixheight, font.pixwidth
After you initialize the fonts and make one the active font, you can display the
Using Fonts from the Graphics Library text on the screen. To display text on the screen after selecting a font: 1. Select a starting position for the text with MOVETO. 2. Optionally, set a text display angle with SETGTEXTROTATION. 3. Send the text to the screen (in the current font) with OUTGTEXT.
Page 4 of 5
MOVETO moves the current graphics point to the pixel coordinates passed to it when it is invoked. This becomes the starting position of the upper-left corner of the first character in the text. SETGTEXTROTATION can set the text's orientation in increments of one-tenth of a degree.
SHOWFONT.F90 Example
The Visual Fortran Sample program SHOWFONT.F90 in the ...\DF98 \SAMPLES\TUTORIAL folder displays text in the fonts available on your system. (Once the screen fills with text, press Enter to display the next screen.) An abbreviated version follows. SHOWFONT calls SETFONT to specify the typeface. MOVETO then establishes the starting point for each text string. The program sends a message of sample text to the screen for each font initialized:
! Abbreviated version of SHOWFONT.F90. USE DFLIB INTEGER(2) grstat, numfonts,indx, curr_height TYPE (xycoord) xyt TYPE (fontinfo) f CHARACTER(6) str ! 5 chars for font num ! (max. is 32767), 1 for 'n' ! Initialization. numfonts=INITIALIZEFONTS( ) IF (numfonts.LE.0) PRINT *,"INITIALIZEFONTS error" IF (GRSTATUS().NE.$GROK) PRINT *,'INITIALIZEFONTS GRSTATUS error.' CALL MOVETO (0,0,xyt) grstat=SETCOLORRGB(#FF0000) grstat=FLOODFILLRGB(0, 0, #00FF00) grstat=SETCOLORRGB(0) ! Get default font height for comparison later. grstat = SETFONT('n1') grstat = GETFONTINFO(f) curr_height = f.pixheight ! Done initializing, start displaying fonts. DO indx=1,numfonts WRITE(str,10)indx grstat=SETFONT(str) IF (grstat.LT.1) THEN CALL OUTGTEXT('SetFont error.') ELSE grstat=GETFONTINFO(f) grstat=SETFONT('n1') CALL OUTGTEXT(f.facename(:len_trim(f.facename))) CALL OUTGTEXT(' ') ! Display font.
Page 5 of 5
Page 1 of 10
Open the Control Panel (available from Settings) Click the Regional Settings icon Choose from the dropdown list of available locales (languages and countries).
When you select a new locale, it becomes the default system locale, and will remain the default locale until you change it. Each locale has a default codepage associated with it, and a default currency, number, and date format. Note: The default codepage does not change when you select a new locale
Using National Language Support Routines until you reboot your computer.
Page 2 of 10
You can change the currency, number, and date format in the International dialog box or the Regional Setting dialog box independently of the locale. The locale determines the character set available to the user. The locale you select becomes the default for the NLS routines described in this section, but the NLS routines allow you to change locales and their parameters from within your programs. These routines are useful for creating original foreign-language programs or different versions of the same program for various international markets. Changes you make to the locale from within a program affect only the program. They do not change the system default settings. The codepage you select, which can be set independently, controls the multibyte (MB routines) character routines described in this section. Only users with special multibyte-character code sets installed on their computers need to use MB routines. The standard code sets all use single-byte character code sets. Note that in Visual Fortran source code, multibyte characters can be used only in character strings and source comments. They cannot be used within variable names or statements. Like program changes to the locale, program changes to codepages affect only the program, not the system defaults. The NLS and MB routines are contained in the library DFNLS.LIB, which consists of DFNLS.MOD and DFNLS.F90. To access the routines, the following statement should be present in any program unit that uses NLS or MB routines:
USE DFNLS
This section includes a discussion of character sets and the NLS library routines:
l l
Single and Multibyte Character Sets National Language Support Library Routines
Page 3 of 10
multibyte-character string can contain a mix of single and double-byte characters. A two-byte character has a lead byte and a trail byte. In a particular multibyte character set, the lead and trail byte values can overlap, and it is then necessary to use the byte's context to determine whether it is a lead or trail byte.
l l
Locale Setting and Inquiry Routines to set locales (local code sets) and inquire about their current settings NLS Formatting Routines to format dates, currency, and numbers Multibyte Character Routines for using multi-byte characters
All of these routines are described in detail in the Language Reference, A to Z Reference. In the descriptions that follow, function and parameter names are given with a mixture of upper- and lowercase letters. This is to make the names easier to understand. You can use any case for these names when writing your applications.
Page 4 of 10
Description Sets the language, country and codepage Retrieves the current language, country and codepage Retrieves requested information about the current local code set Returns all the languages and country combinations supported by the system Returns all the supported codepages on the system Changes the codepage for the current console Returns the codepage number for the system (Window) codepage or the console codepage
NLSGetLocaleInfo
Function
NLSEnumLocales
Function
As an example:
USE DFNLS INTEGER(4) strlen, status CHARACTER(40) str strlen = NLSGetLocaleInfo(NLS$LI_SDAYNAME1, print *, str ! prints Monday strlen = NLSGetLocaleInfo(NLS$LI_SDAYNAME2, print *, str ! prints Tuesday strlen = NLSGetLocaleInfo(NLS$LI_SDAYNAME3, print *, str ! prints Wednesday ! Change locale to Spanish, Mexico status = NLSSetLocale("Spanish", "Mexico") strlen = NLSGetLocaleInfo(NLS$LI_SDAYNAME1, print *, str ! prints lunes strlen = NLSGetLocaleInfo(NLS$LI_SDAYNAME2, print *, str ! prints martes strlen = NLSGetLocaleInfo(NLS$LI_SDAYNAME3, print *, str ! prints mircoles END str) str) str)
Page 5 of 10
NLSFormatDate
Function
NLSFormatNumber
Function
NLSFormatTime
Function
As an example:
USE DFNLS INTEGER(4) strlen, status CHARACTER(40) str strlen = NLSFormatTime(str) print *, str ! prints 11:42:24 AM strlen = NLSFormatDate(str, flags= NLS$LongDate) print *, str ! prints Friday, July 14, 2000 status = NLSSetLocale ("Spanish", "Mexico") strlen = NLSFormatTime(str) print *, str ! prints 11:42:24
Page 6 of 10
MBCS Inquiry Routines MBCS Conversion Routines MBCS Fortran Equivalent Routines Standard Fortran Routines That Handle MBCS Characters
Page 7 of 10
Returns the string position of the first byte of the multibyte character immediately after the given string position Returns the string position of the first byte of the multibyte character immediately before the given string position Performs a context sensitive test to determine whether a given byte in a character string is a lead byte
MBPrev
Function
MBStrLead
Function
As an example:
USE DFNLS CHARACTER(4) str INTEGER status status = NLSSetLocale ("Japan") str = " , " " PRINT '(1X,''String by char = '',\)' DO i = 1, len(str) PRINT '(A2,\)',str(i:i) END DO PRINT '(/,1X,''MBLead = '',\)' DO i = 1, len(str) PRINT '(L2,\)',mblead(str(i:i)) END DO PRINT '(/,1X,''String as whole = '',A,\)',str PRINT '(/,1X,''MBStrLead = '',\)' DO i = 1, len(str) PRINT '(L1,\)',MBStrLead(str,i) END DO END
Two convert Japan Industry Standard characters to Microsoft Kanji characters or vice versa. Two convert between a codepage multibyte character string and a Unicode string.
Page 8 of 10
MBCS Conversion Routines Name MBConvertMBToUnicode Procedure Type Function Description Converts a character string from a multibyte codepage to a Unicode string Converts a Unicode string to a multibyte character string of the current codepage Converts a Japan Industry Standard (JIS) character to a Microsoft Kanji (Shift JIS or JMS) character Converts a Microsoft Kanji (Shift JIS or JMS) character to a Japan Industry Standard (JIS) character
MBConvertUnicodeToMB
Function
MBJISToJMS
Function
MBJMSToJIS
Function
MBINDEX
Function
Functions
Page 9 of 10
Same as SCAN except that multibyte characters can be included in its arguments Same as VERIFY except that multibyte characters can be included in its arguments
MBVERIFY
Function
The following example is included in Visual Fortran Samples in the ...\DF98 \SAMPLES\TUTORIAL folder as MBCOMP.FOR:
USE DFNLS INTEGER(4) i, len(7), infotype(7) CHARACTER(10) str(7) LOGICAL(4) log4 data infotype / NLS$LI_SDAYNAME1, NLS$LI_SDAYNAME2, & & NLS$LI_SDAYNAME3, NLS$LI_SDAYNAME4, & & NLS$LI_SDAYNAME5, NLS$LI_SDAYNAME6, & & NLS$LI_SDAYNAME7 / WRITE(*,*) 'NLSGetLocaleInfo' WRITE(*,*) '----------------' WRITE(*,*) ' ' WRITE(*,*) 'Getting the names of the days of the week...' DO i = 1, 7 len(i) = NLSGetLocaleInfo(infotype(i), str(i)) WRITE(*, 11) 'len/str/hex = ', len(i), str(i), str(i) END DO 11 FORMAT (1X, A, I2, 2X, A10, 2X, '[', Z20, ']') WRITE(*,*) ' ' WRITE(*,*) 'Lexically comparing the names of the days...' DO i = 1, 6 log4 = MBLGE(str(i), str(i+1), NLS$IgnoreCase) WRITE(*, 12) 'Is day ', i, ' GT day ', i+1, '? Answer = ', log4 END DO 12 FORMAT (1X, A, I1, A, I1, A, L1) WRITE(*,*) ' ' WRITE(*,*) 'Done.' END
This code produces the following output when the locale is Japan:
Page 10 of 10
Some routines work with the computer collating sequence to return a character in a certain position in the sequence or the position in the sequence of a certain character. These functions are not dependent on a particular collating sequence. (You should note, however, that elsewhere in this manual the ASCII collating sequence is mentioned in reference to these functions.) Such functions use position and c values between 0 and 255 (inclusive) and include:
l l l l
Because Fortran uses character lengths instead of NULLs to indicate the length of a string, some functions work solely from the length of the string, and not with the contents of the string. This function works as usual on strings containing MBCS characters, and include: REPEAT ( string, ncopies )
Portability Library
Page 1 of 11
Portability Library
Visual Fortran includes functions and subroutines that ease porting of code from a different platform to a PC, or allow you to write code on a PC that is compatible with other platforms. Frequently used functions are included in a module called DFPORT. This chapter describes how to use the portability module, and describes routines available in the following categories:
l l l l l l l l l l
Using the Portability Library Routines for Information Retrieval Device and Directory Information Routines Process Control Routines Numeric Routines Input and Output With Portability Routines Date and Time Routines Error Handling Routines Miscellaneous String and Sorting Routines Other Compatibility Routines
When writing new code, use Fortran 95/90 contains intrinsic procedures whenever possible (for portability and performance reasons). Fortran 95/90 contains intrinsic procedures for many of the portability functions. The portability routines are extensions to the Fortran 95 standard.
Add the statement USE DFPORT to your program. This USE statement includes the DFPORT module. Call portability routines using the correct parameters and return value.
The portability library DFPORT.LIB library is passed to the linker by default during linking. To prevent DFPORT.LIB library from being passed to the linker, specify the /fpscomp:nolibs option. Using the DFPORT module provides interface blocks and parameter definitions for the routines, as well as compiler verification of calls. Some routines in this library can be called with different sets of arguments, and sometimes even as a function instead of a subroutine. In these cases, the arguments and calling mechanism determine the meaning of the routine. The DFPORT module contains generic interface blocks that give procedure definitions
Page 2 of 11
Group, user, and process ID are INTEGER(4) variables. Login name and host name are character variables. The functions GETGID and GETUID are provided for portability, but always return 1. IARGC is best used with GETARG. GETARG, which returns command line arguments, is available in the standard Visual Fortran library; you do not have to specify USE DFPORT in your program unit. For more information:
l
Portability Library
Page 3 of 11
You can retrieve information about devices, directories, and files with the functions listed below. File names can be long file names or UNC file names. A forward slash in a path name is treated as a backslash. All path names can contain drive specifications. Portability Routine CHDIR FSTAT GETCWD RENAME STAT, LSTAT UNLINK Description Changes the current working directory Returns information about a logical file unit Returns the current working directory path name Renames a file Returns information about a named file Removes a directory entry from the path
Standard Fortran 90 provides the INQUIRE statement, which returns detailed file information either by file name or unit number. Use INQUIRE as an equivalent to FSTAT, LSTAT or STAT. LSTAT and STAT return the same information; STAT is the preferred function.
Portability Library SIGNAL SLEEP SYSTEM Changes the action for a signal
Page 4 of 11
Suspends program execution for a specified number of seconds Executes a command in a separate shell
Note that when you use SYSTEM, commands are run in a separate shell. Defaults set with the SYSTEM function, such as current working directory or environment variables, do not affect the environment the calling program runs in. The portability library does not include the FORK routine. On U*X systems, FORK creates a duplicate image of the parent process. Child and parent processes each have their own copies of resources, and become independent from one another. With Windows operating systems, you can create a child process (called a thread), but both parent and child processes share the same address space and share system resources. If you need to create another process, use the CreateProcess Win32 API routine. For information on how to implement threading, see Creating Multithread Applications.
Numeric Routines
Numeric routines are available for calculating Bessel functions, data type conversion, and generating random numbers: Portability Routine BESJ0, BESJ1, BESJN, BESY0, BESY1, BESYN DBESJ0, DBESJ1, DBESJN, DBESY0, DBESY1, DBESYN LONG SHORT Description Computes the single precision values of Bessel functions of the first and second kind of orders 1, 2, and n, respectively Computes the double-precision values of Bessel functions of the first and second kind of orders 1, 2, and n, respectively Converts an INTEGER(2) variable to an INTEGER (4) type Converts an INTEGER(4) variable to an INTEGER (2) type
Page 5 of 11 Returns a positive integer in the range 0 through (2**31)-1, or (2**15)-1 if called without an argument Returns random values in the range 0 through 1.0 Returns random values in the range 0 through 1.0 Returns random values in the range 0 through 1.0 Seeds the random number generator used with IRAND and RAND. Perform bit level clear, set, and test for integers
Some of these functions have equivalents in standard Fortran 95/90. Data object conversion can be accomplished by using the INT intrinsic function instead of LONG or SHORT. The intrinsic subroutines RANDOM_NUMBER and RANDOM_SEED perform the same functions as the random number functions listed in the previous table. Other bit manipulation functions such as AND, XOR, OR, LSHIFT, and RSHIFT are intrinsic functions. You do not need the DFPORT module to access them. Standard Fortran 95/90 includes many bit operation procedures; these are listed in the Bit Operation Procedures table in the Language Reference.
When used with direct files, after an FSEEK, GETC, or PUTC operation, the record number is the number of the next whole record. Any subsequent normal Fortran I/O to that unit occurs at the next whole record. For example, if you seek to absolute location 1 of a file whose record length is 10, the NEXTREC returned by an inquire would be 2. If you seek to absolute location 10, NEXTREC would still return 2. On units with CARRIAGECONTROL='FORTRAN' (the default), PUTC and FPUTC characters are treated as carriage control characters if they appear in column 1.
Portability Library
l
Page 6 of 11
On sequentially formatted units, the C string "\n"c, which represents the carriage return/line feed escape sequence, is written as CHAR(13) (carriage return) and CHAR(10) (line feed), instead of just line feed, or CHAR(10). On input, the sequence 13 followed by 10 is returned as just 10. (The length of character string "\n"c is 1 character, whose ASCII value, indicated by ICHAR('\n'c), is 10.) Reading and writing is in a raw form for direct files. Separators between records can be read and overwritten. Therefore, be careful if you continue using the file as a direct file.
I/O errors arising from the use of these routines result in a Visual Fortran runtime error. Portability Routine ACCESS CHMOD FGETC FLUSH FPUTC FSEEK FTELL GETC PUTC Description Checks a file for accessibility according to mode Changes file attributes Reads a character from an external unit Flushes the buffer for an external unit to its associated file Writes a character to an external unit Repositions a file on an external unit Returns the offset, in bytes, from the beginning of the file Reads a character from unit 5 Writes a character to unit 6
All path names can include drive specifications, forward slashes, or backslashes. Some portability file I/O routines have equivalents in standard Fortran 95/90. For example, you could use the ACCESS function to check a file specified by name for accessibility according to mode. It tests a file for read, write, or execute permission, as well as checking to see if the file exists. It works on the file attributes as they exist on disk, not as a program's OPEN statement specifies them. Instead of ACCESS, you can use the INQUIRE statement with the ACTION parameter to check for similar information. (The ACCESS function always
Portability Library
Page 7 of 11
returns 0 for read permission on FAT files, meaning that all files have read permission.)
CTIME DATE
"Wed Feb 24 19:11:00 1999" As a subroutine: "24-Feb-99" As a function: "02/24/99" (/0.0, 0.0/) (Actual results depend on the program and the system) (/0.0, 0.0/) (Actual results depend on the program and the system) "Wed Feb 24 19:11:00 1999" (/0,12,03,24,2,97,3,55,0/)
DTIME
ETIME
FDATE GMTIME
Portability Library IDATE Current date either as one 3-element array or three scalar parameters (month, day, year) Current time as a 3-element array (hour, minute, second) Current date as an 8character string with the Julian date Local time as a 9-element integer array Number of seconds since 00:00:00 GMT, Jan 1, 1970 The number of seconds since midnight, less the value of its argument As a subroutine, returns the time formatted as hh:mm:ss As a function, returns the time in seconds since midnight GMT Jan 1, 1970 TIMEF The number of seconds since the first time this function was called (or zero) 0.0
ITIME
(/7,11,00/)
JDATE
"99055 "
LTIME RTC
(/0,11,7,24,2,99,3,55,0/) 762145860
SECNDS
0.00
TIME
TIME and DATE are available as either a function or subroutine. Because of the name duplication, if your programs do not include the USE DFPORT statement, each separately compiled program unit can use only one of these versions. For example, if a program calls the subroutine TIME once, it cannot also use TIME as a function. Standard Fortran 95/90 includes date and time intrinsic subroutines. For more information, see DATE_AND_TIME in the Language Reference.
Portability Library
Page 9 of 11
The following routines are available for detecting and reporting errors: Portability Routine IERRNO GERROR PERROR Description Returns the last error code Returns the IERRNO error code as a string variable Sends an error message, preceded by a string, for the last error detected
IERRNO error codes are analogous to errno on U*X systems. The DFPORT module provides parameter definitions for many of U*X's errno names, found typically in errno.h on U*X systems. IERRNO is updated only when an error occurs. For example, if a call to the GETC function results in an error, but two subsequent calls to PUTC succeed, a call to IERRNO returns the error for the GETC call. Examine IERRNO immediately after returning from one of the portability library routines. Other standard Fortran 90 routines might also change the value to an undefined value. If your application uses multithreading, remember that IERRNO is set on a perthread basis.
For information on equivalent standard Fortran 95/90 character procedures, see Character Procedures in the Language Reference.
Portability Library
Page 10 of 11
Visual Fortran does not support certain other functions, such as: Routine CMVGM, CMVGN, CMVGP, CMVGT, CMVGZ FORK LINK SYMLNK Description Conditional merge Similar Visual Fortran Functionality MERGE intrinsic function
Creates an identical process Creates a hard link between two files Creates a symbolic link between two files
Page 11 of 11
Replace conditional merge routines with the standard Fortran 95/90 intrinsic MERGE routine, using the following arguments: Routine CVMGP(tsrc, fsrc, mask) CVMGM(tsrc, fsrc, mask) CVMGZ(tsrc, fsrc, mask) CVMGN(tsrc, fsrc, mask) CVMGT(tsrc, fsrc, mask) Fortran 95/90 Replacement MERGE(tsrc, fsrc, mask >= 0) MERGE(tsrc, fsrc, mask < 0) MERGE(tsrc, fsrc, mask = 0) MERGE(tsrc, fsrc, mask /= 0) MERGE(tsrc, fsrc, mask = .TRUE.)
There is no analogy to U*X file system links or soft links under Windows. There is also no analogy to the U*X FORK routine, since FORK creates a duplicate image of the parent process which is independent from the parent process. With Windows operating systems, both parent and child processes share the same address space and share system resources. For more information on creating child processes, see Creating Multithread Applications.
Page 1 of 28
Devices and Files I/O Hardware Using the Console Using the Serial I/O Port Routines
Logical Devices, which describes logical unit specifiers, external files, and internal files. Files, which describes the types of files, file access, record types, and Microsoft Fortran PowerStation compatible files.
Logical Devices
Every file, internal or external, is associated with a logical device. You identify the logical device associated with a file by a unit specifier (UNIT=). The unit specifier for an internal file is the name of the character variable associated with it. The unit specifier for an external file is either a number you assign with the OPEN statement, a number preconnected as a unit specifier to a device, or an asterisk (*).
Page 2 of 28
External unit specifiers that are preconnected to certain devices do not have to be opened. External units that you connect are disconnected when program execution terminates or when the unit is closed by a CLOSE statement. A unit must not be connected to more than one file at a time, and a file must not be connected to more than one unit at a time. You can OPEN an already opened file but only to change some of the I/O options for the connection, not to connect an already opened file or unit to a different unit or file. You must use a unit specifier for all I/O statements, except in the following three cases:
l
ACCEPT, which always reads from standard input, unless the FOR_ACCEPT environment variable is defined. INQUIRE by file, which specifies the filename, rather than the unit with which the file is associated PRINT, which always writes to standard output, unless the FOR_PRINT environment variable is defined. READ statements that contain only an I/O list and format specifier, which read from standard input (UNIT=5), unless the FOR_READ environment variable is defined. WRITE statements that contain only an I/O list and format specifier, which write to standard output, unless the FOR_PRINT environment variable is defined. TYPE, which always writes to standard output, unless the FOR_TYPE environment variable is defined.
External Files A unit specifier associated with an external file must be either an integer expression or an asterisk (*). The integer expression must be in the range 0 (zero) to a maximum value of 2,147,483,640. The following example connects the external file UNDAMP.DAT to unit 10 and writes to it:
OPEN (UNIT = 10, FILE = 'undamp.dat') WRITE (10, '(A18,\)') ' Undamped Motion:'
The asterisk (*) unit specifier specifies the keyboard when reading and the screen when writing. The following example uses the asterisk specifier to write to the screen:
WRITE (*, '(1X, A30,\)') ' Write this to the screen.'
Visual Fortran has four units preconnected to external files (devices), as shown in the following table.
Page 3 of 28
Description Always represents the keyboard and screen (unless the appropriate environment variable is defined, such as FOR_READ). Initially represents the screen (unless FORT0 is defined) Initially represents the keyboard (unless FORT5 is defined) Initially represents the screen (unless FORT6 is defined)
0 5 6
The asterisk (*) specifier is the only unit specifier that cannot be reconnected to another file, and attempting to close this unit causes a compile-time error. Units 0, 5, and 6, however, can be connected to any file with the OPEN statement. If you close unit 0, 5, or 6, it is automatically reconnected to its respective device the next time an I/O statement attempts to use that unit. When you omit the file name in the OPEN statement or use an implicit OPEN, you can define the environment variable FORTn to specify the file name for a particular unit number (n) (except when the compiler option /fpscomp:filesfromcmd is not specified). For example, if you want unit 6 to write to a file instead of standard output, set the environment variable FORT6 to the path and filename to be used before you run the program. The following example writes to the preconnected unit 6 (the screen), then reconnects unit 6 to an external file and writes to it, and finally reconnects unit 6 to the screen and writes to it:
REAL a, b Write to the screen (preconnected unit 6). WRITE(6, '('' This is unit 6'')') ! Use the OPEN statement to connect unit 6 ! to an external file named 'COSINES'. OPEN (UNIT = 6, FILE = 'COSINES', STATUS = 'NEW') DO a = 0.1, 6.3, 0.1 b = COS (a) ! Write to the file 'COSINES'. WRITE (6, 100) a, b 100 FORMAT (F3.1, F5.2) END DO ! Close it. CLOSE (6) ! Reconnect unit 6 to the screen, by writing to it. WRITE(6,' ('' Cosines completed'')') END !
Internal Files
Page 4 of 28
The unit specifier associated with an internal file is a character string or character array. There are two types of internal files:
l
An internal file that is a character variable, character array element, or noncharacter array element that has exactly one record, which is the same length as the variable, array element, or noncharacter array element. An internal file that is a character array, a character derived type, or a noncharacter array that is a sequence of elements, each of which is a record. The order of records is the same as the order of array elements or type elements, and the record length is the length of one array element or the length of the derived-type element.
Use only formatted I/O, including I/O formatted with a format specification and list-directed I/O. (List-directed I/O is treated as sequential formatted I/O.) Namelist formatting is not allowed. If the character variable is an allocatable array or array part of an allocatable array, the array must be allocated before use as an internal file. If the character variable is a pointer, it must be associated with a target. Use only READ and WRITE statements. You cannot use file connection (OPEN, CLOSE), file positioning (REWIND, BACKSPACE) or file inquiry (INQUIRE) statements with internal files.
You can read and write internal files with FORMAT I/O statements or listdirected I/O statements exactly as you can external files. Before an I/O statement is executed, internal files are positioned at the beginning, before the first record. With internal files, you can use the formatting capabilities of the I/O system to convert values between external character representations and Fortran internal memory representations. That is, reading from an internal file converts the ASCII representations into numeric, logical, or character representations, and writing to an internal file converts these representations into their ASCII representations. This feature makes it possible to read a string of characters without knowing its exact format, examine the string, and interpret its contents. It also makes it possible, as in dialog boxes, for the user to enter a string and for your application to interpret it as a number. If less than an entire record is written to an internal file, the rest of the record is filled with blanks. In the following example, str and fname specify internal files:
Page 5 of 28
CHARACTER(10) str INTEGER n1, n2, n3 CHARACTER(14) fname INTEGER i str = " 1 2 3" ! List-directed READ sets n1 = 1, n2 = 2, n3 = 3. READ(str, *) n1, n2, n3 i = 4 ! Formatted WRITE sets fname = 'FM004.DAT'. WRITE (fname, 200) i 200 FORMAT ('FM', I3.3, '.DAT')
Files
File organization refers to the way records are physically arranged on a storage device. Record type refers to whether records in a file are all the same length, are of varying length, or use other conventions to define where one record ends and another begins. Record access refers to the method used to read records from or write records to a file, regardless of its organization. The way a file is organized does not necessarily imply the way in which the records within that file will be accessed. Fortran supports two kinds of file organizations: sequential and relative. The organization of a file is specified by means of the ORGANIZATION keyword in the OPEN statement. Relative files must be stored on disk. However, sequential files can be stored on either magnetic tape or disk. Other peripheral devices, such as terminals, pipes, card readers, and line printers, are treated as sequential files. A sequentially organized file consists of records arranged in the sequence in which they are written to the file (the first record written is the first record in the file, the second record written is the second record in the file, and so on). As a result, records can be added only at the end of the file. Attempting to add records at someplace other than the end of the file will result in the file begin truncated at the end of the record just written. Within a relative file are numbered positions, called cells. These cells are of fixed equal length and are consecutively numbered from 1 to n, where 1 is the first cell, and n is the last available cell in the file. Each cell either contains a single record or is empty. Records in a relative file are accessed according to cell number. A cell number is a record's relative record number; its location relative to the beginning of the file. By specifying relative record numbers, you can directly retrieve, add, or delete records regardless of their locations. (Detecting deleted records is only available if you specified the /vms option when the program was compiled. For information, see the /vms option.)
Page 6 of 28
Fortran supports two methods of file access (sequential and direct) and three kinds of file structure (formatted, unformatted, and binary). Sequential-access and direct-access files can have any of the three file structures. The following kinds of files are possible:
l l l l l l
Formatted Sequential Formatted Direct Unformatted Sequential Unformatted Direct Binary Sequential Binary Direct
Each kind of file has advantages and the best choice depends on the application you are developing:
l
Formatted Files You create a formatted file by opening it with the FORM='FORMATTED' option, or by omitting the FORM parameter when creating a sequential file. The records of a formatted file are stored as ASCII characters; numbers that would otherwise be stored in binary form are converted to ASCII format. Each record ends with the ASCII carriage return (CR) and line feed (LF) characters. If you need to view a data file's contents, use a formatted file. You can load a formatted file into a text editor and read its contents directly, that is, the numbers would look like numbers and the strings like character strings, whereas an unformatted or binary file looks like a set of hexadecimal characters.
Unformatted Files You create an unformatted file by opening it with the FORM='UNFORMATTED' option, or by omitting the FORM parameter when creating a direct-access file. An unformatted file is a series of records composed of physical blocks. Each record contains a sequence of values stored in a representation that is close to that used in program memory. Little conversion is required during input/output. The lack of formatting makes these files quicker to access and more compact than files that store the same information in a formatted form. However, if the files contain numbers, you will not be able to read them with a text editor.
Binary Files
Page 7 of 28
You create a binary file by specifying FORM='BINARY'. Binary files are the most compact, and good for storing large amounts of data.
l
Sequential-Access Files Data in sequential files must be accessed in order, one record after the other (unless you change your position in the file with the REWIND or BACKSPACE statements). Some methods of I/O are possible only with sequential files, including nonadvancing I/O, list-directed I/O, and namelist I/O. Internal files also must be sequential files. You must use sequential access for files associated with sequential devices. A sequential device is a physical storage device that does not allow explicit motion (other than reading or writing). The keyboard, screen, and printer are all sequential devices.
Direct-Access Files Data in direct-access files can be read or written to in any order. Records are numbered sequentially, starting with record number 1. All records have the length specified by the RECL option in the OPEN statement. Data in direct files is accessed by specifying the record you want within the file. If you need random access I/O, use direct-access files. A common example of a random-access application is a database.
All files are composed of records. Each record is one entry in the file. It can be a line from a terminal or a logical record on a magnetic tape or disk file. All records within one file are of the same type. In Fortran, the number of bytes written to a record must be less than or equal to the record length. One record is written for each unformatted READ or WRITE statement. A formatted READ or WRITE statement can transfer more than one record using the slash (/) edit descriptor. For binary files, a single READ or WRITE statement reads or writes as many records as needed to accommodate the number of bytes being transferred. On output, incomplete formatted records are padded with spaces. Incomplete unformatted and binary records are padded with undefined bytes (zeros). The remainder of this section contains information about:
l l
Record Types Microsoft Fortran PowerStation Compatible Files (when /fpscomp was specified during compilation)
Record Types
Page 8 of 28
An I/O record is a collection of data items, called fields, that are logically related and are processed as a unit. The record type refers to the convention for storing fields in records. The record type of the data within a file is not maintained as an attribute of the file. The results of using a record type other than the one used to create the file are indeterminate. If you omit /fpscomp:ioformat during compilation, the following six record types are available: Record Type Fixed-length Variablelength Segmented Available File Organizations and Portability Considerations Relative or sequential file organizations. Sequential file organization only. The variable-length record type is generally the most portable record type across multivendor platforms. Sequential file organization only and only for unformatted sequential access. The segmented record type is unique to Compaq Fortran and should not be used for portability with programs written in languages other than Fortran or for places where Compaq Fortran is not used. However, because the segmented record type is unique to Compaq Fortran products, formatted data in segmented files can be ported across Compaq Fortran platforms. Sequential file organization only.
Stream (uses no record delimiter) Stream_CR (uses CR as record delimiter) Stream_LF (uses CR and LF as record delimiter)
Fixed-Length Records
Page 9 of 28
When you specify fixed-length records, you are specifying that all records in the file contain the same number of bytes. When you open a file that is to contain fixed-length records, you must specify the record size using the RECL keyword. A sequentially organized opened file for direct access must contain fixed-length records, to allow the record position in the file to be computed correctly. For relative files, the layout and overhead of fixed-length records depends upon whether the program accessing the file was compiled using the /vms option or whether the /vms option was omitted:
l
For relative files where the /vms option was omitted (the default), each record has no control information. For relative files where the /vms option was specified, each record has one byte of control information at the beginning of the record.
Variable-Length Records Variable-length records can contain any number of bytes, up to a specified maximum record length, and only apply to sequential files. These records are generally prefixed and suffixed by four bytes of control information containing count fields. The 4-byte integer value stored in each count field indicates the number of data bytes (excluding overhead bytes) in that particular variablelength record. The record layout of variable-length records appears below.
The count field of a variable-length record is available when you read the record
Page 10 of 28
by issuing a READ statement with a Q format descriptor. You can then use the count field information to determine how many bytes should be in the associated I/O list. Files written with variable-length records by Compaq Fortran programs usually cannot be accessed as text files. Instead, use the Stream_LF record format for text files with records of varying length. Segmented Records A segmented record is a single logical record consisting of one or more variablelength, unformatted records in a sequentially organized disk file. Unformatted data written to sequentially organized files using sequential access is stored as segmented records by default. Segmented records are useful when you want to write exceptionally long records but cannot or do not wish to define one long variable-length record, perhaps because virtual memory limitations can prevent program execution. By using smaller, segmented records, you reduce the chance of problems caused by virtual memory limitations on systems on which the program may execute. For disk files, the segmented record is a single logical record that consists of one or more segments. Each segment is a physical record. A segmented (logical) record can exceed the absolute maximum record length (2.14 billion bytes), but each segment (physical record) individually cannot exceed the maximum record length. To access an unformatted sequential file that contains segmented records, specify FORM='UNFORMATTED' and RECORDTYPE='SEGMENTED' when you open the file. Otherwise, the file may be processed erroneously. As shown below, the layout of segmented records consists of 4 bytes of control information followed by the user data.
The control information consists of a 2-byte integer record length count (includes the 2 bytes used by the segment identifier), followed by a 2-byte integer segment identifier that identifies this segment as one of the following:
l l l
The first segment (equals 1) The last segment (equals 2) The only segment (equals 3)
Page 11 of 28
One of the segments between the first and last segments (equals 0)
If the specified record length is an odd number, the user data will be padded with a single blank (1 byte), but this extra byte is not added to the 2-byte integer record length count. Stream File Data A stream file is not grouped into records and contains no control information. Stream files are used with CARRIAGECONTROL='NONE' and contain character or binary data that is read or written only to the extent of the variables specified on the input or output statement. The layout of a stream file appears below.
Stream_CR Records A stream_CR record is a variable-length record whose length is indicated by explicit record terminators embedded in the data, not by a count. These terminators are automatically added when you write records to a stream-type file, and they are removed when you read records. Stream_CR files use only a carriage-return as the terminator, so Stream_CR files must not contain embedded carriage-return characters. The layout of stream_CR records appears below.
Stream_LF Records A stream_LF record is a variable-length record whose length is indicated by explicit record terminators embedded in the data, not by a count. These terminators are automatically added when you write records to a stream-type file, and they are removed when you read records. Stream_LF files use a carriage return followed by line-feed (new line) as the terminator, so Stream_LF files must not contain embedded line-feed (new line) characters. This is the usual operating system text file record type. The layout of stream_LF records appears below.
Page 12 of 28
Formatted Sequential Files A formatted sequential file is a series of formatted records written sequentially and read in the order in which they appear in the file. Records can vary in length and can be empty. They are separated by carriage return (0D) and line feed (0A) characters as shown in the following figure. Formatted Records in a Formatted Sequential File
An example of a program writing three records to a formatted sequential file is given below. The resulting file is shown in the following figure.
OPEN (3, FILE='FSEQ') ! FSEQ is a formatted sequential file by default. WRITE (3, '(A, I3)') 'RECORD', 1 WRITE (3, '()') WRITE (3, '(A11)') 'The 3rd One' CLOSE (3) END
Page 13 of 28
In a formatted direct file, all of the records are the same length and can be written or read in any order. The record size is specified with the RECL option in an OPEN statement and should be equal to or greater than the number of bytes in the longest record. The carriage return (CR) and line feed (LF) characters are record separators and are not included in the RECL value. Once a direct-access record has been written, you cannot delete it, but you can rewrite it. During output to a formatted direct file, if data does not completely fill a record, the compiler pads the remaining portion of the record with blank spaces. The blanks ensure that the file contains only completely filled records, all of the same length. During input, the compiler by default also pads the input if the input list and format require more data than the record contains. You can override the default blank padding on input by setting PAD='NO' in the OPEN statement for the file. If PAD='NO', the input record must contain the amount of data indicated by the input list and format specification. Otherwise, an error occurs. PAD='NO' has no effect on output. An example of a program writing two records, record one and record three, to a formatted direct file is given below. The result is shown in the following figure.
OPEN (3,FILE='FDIR', FORM='FORMATTED', ACCESS='DIRECT',RECL=10) WRITE (3, '(A10)', REC=1) 'RECORD ONE' WRITE (3, '(I5)', REC=3) 30303 CLOSE (3) END
Page 14 of 28
Unformatted sequential files are organized slightly differently on different platforms. This section describes unformatted sequential files created by Visual Fortran when the /fpscomp option (such as /fpscomp: (ioformat,general)) was specified. If you are accessing files from another platform that organizes them differently, see Converting Unformatted Numeric Data or you can use the conversion utility in the ...\DF98 \SAMPLES\TUTORIAL folder called UNFSEQ.F90. The records in an unformatted sequential file can vary in length. Unformatted sequential files are organized in chunks of 130 bytes or less called physical blocks. Each physical block consists of the data you send to the file (up to 128 bytes) plus two 1-byte "length bytes" inserted by the compiler. The length bytes indicate where each record begins and ends. A logical record refers to an unformatted record that contains one or more physical blocks. (See the following figure.) Logical records can be as big as you want; the compiler will use as many physical blocks as necessary. When you create a logical record consisting of more than one physical block, the compiler sets the length byte to 129 to indicate that the data in the current physical block continues on into the next physical block. For example, if you write 140 bytes of data, the logical record has the structure shown in the following figure. Logical Record in Unformatted Sequential File
The first and last bytes in an unformatted sequential file are reserved; the first contains a value of 75, and the last holds a value of 130. Fortran uses these bytes for error checking and end-of-file references. The following program creates the unformatted sequential file shown in the following figure:
! ! ! Note: The file is sequential by default -1 is FF FF FF FF hexadecimal. CHARACTER xyz(3) INTEGER(4) idata(35) DATA idata /35 * -1/, xyz /'x', 'y', 'z'/ ! ! Open the file and write out a 140-byte record: ! 128 bytes (block) + 12 bytes = 140 for IDATA, then 3 bytes for XYZ.
Page 15 of 28
Unformatted Direct Files An unformatted direct file is a series of unformatted records. You can write or read the records in any order you choose. All records have the same length, given by the RECL specifier in an OPEN statement. No delimiting bytes separate records or otherwise indicate record structure. You can write a partial record to an unformatted direct file. Visual Fortran pads these records to the fixed record length with ASCII NULL characters. Unwritten records in the file contain undefined data. The following program creates the sample unformatted direct file shown in the following figure:
OPEN (3, FILE='UFDIR', RECL=10,& & FORM = 'UNFORMATTED', ACCESS = 'DIRECT') WRITE (3, REC=3) .TRUE., 'abcdef' WRITE (3, REC=1) 2049 CLOSE (3) END
Page 16 of 28
Binary Sequential Files A binary sequential file is a series of values written and read in the same order and stored as binary numbers. No record boundaries exist, and no special bytes indicate file structure. Data is read and written without changes in form or length. For any I/O data item, the sequence of bytes in memory is the sequence of bytes in the file. The next program creates the binary sequential file shown in the following figure:
! ! ! NOTE: 07 is the bell character Sequential is assumed by default. INTEGER(1) bells(4) CHARACTER(4) wys(3) CHARACTER(4) cvar DATA bells /4*7/ DATA cvar /' is '/,wys /'What',' you',' see'/ OPEN (3, FILE='BSEQ',FORM='BINARY') WRITE (3) wys, cvar WRITE (3) 'what ', 'you get!' WRITE (3) bells CLOSE (3) END
Binary Direct Files A binary direct file stores records as a series of binary numbers, accessible in any order. Each record in the file has the same length, as specified by the RECL argument to the OPEN statement. You can write partial records to binary direct files; any unused portion of the record will contain undefined data. A single read or write operation can transfer more data than a record contains by continuing the operation into the next records of the file. Performing such an operation on an unformatted direct file would cause an error. Valid I/O operations for unformatted direct files produce identical results when they are performed on binary direct files, provided the operations do not depend on zero padding in partial records.
Page 17 of 28
The following program creates the binary direct file shown in the following figure:
OPEN (3, FILE='BDIR',RECL=10,FORM='BINARY',ACCESS='DIRECT') WRITE (3, REC=1) 'abcdefghijklmno' WRITE (3) 4,5 WRITE (3, REC=4) 'pq' CLOSE (3) END
I/O Hardware
Most of your hardware configuration and setup is done through your computer's operating system. To connect and communicate with your printer, for example, you should read your system and printer manuals. This section describes how Visual Fortran refers to physical devices and shortcuts for printing text and graphics from the Microsoft visual development environment. For more information:
l l
Printing
The simplest way to print a file while you are in the visual development environment is to choose File/Print from the file menu. You are prompted for the file name and the file is printed on the printer connected to your computer. You can also print files with the extension .F90, .FOR, .FD, .FI, or .RC by dragging the file from Windows Explorer and dropping it on the Print Manager icon. If you have drawn graphics on the screen and want to print the screen, the simplest way is to press the key combination ALT+PRINT SCREEN. This copies the
Page 18 of 28
active window (the one with graphical focus) onto the Clipboard. (If you only press PRINT SCREEN, it prints your entire screen including any background applications.) Once you have copied your screen onto the Clipboard, open Paintbrush and select Edit/Paste to paste the image from the Clipboard into Paintbrush, then select File/Print to print it to the printer connected to your computer. You can also save the image as a bitmap (.BMP) file.
Physical Devices
I/O statements that do not refer to a specific file or I/O device read from standard input and write to standard output. Standard input is the keyboard, and standard output is the screen (console). To perform input and output on a physical device other than the keyboard or screen, you specify the device name as the filename to be read from or written to. Some physical device names are determined by the host operating system; others are recognized by Visual Fortran. Extensions on most device names are ignored. Filenames for Device I/O Device CON PRN COM1 COM2 COM3 COM4 LPT1 LPT2 LPT3 LPT4 NUL AUX Description Console (standard output) Printer Serial port #1 Serial port #2 Serial port #3 Serial port #4 Parallel Port #1 Parallel Port #2 Parallel Port #3 Parallel Port #4 NULL device. Discards all output; contains no input Serial port #1
Files, Devices, and I/O Hardware LINE 1 USER 1 ERR 1 CONOUT$ CONIN$
1
Page 19 of 28
Serial port #1 Standard output Standard error Standard output Standard input
If you use one of these names with an extensionfor example, LINE.TXT Fortran will write to a file rather than to the device. Examples of opening physical devices as units are:
OPEN (UNIT = 4, FILE = 'PRN') OPEN (UNIT = 7, FILE = 'LPT2', ERR = 100)
The actual console window that shows the characters on the screen. The console buffer that contains the characters to be displayed.
If the console screen buffer is bigger than the console window, scroll bars are provided automatically. The size of the console screen buffer must be larger (or equal to) the size of the console window (if not, an error occurs). For applications that need to display more than a few hundred lines of text, the
Page 20 of 28
ability to scroll quickly through the text is important. Also, the maximum size of the console window buffer differs between Windows operating systems:
l
On a Windows 98, Windows Me, or Windows 95 system, the number of character lines is limited to about 40-50 pages of output with 132character lines. On a Windows NT or Windows 2000 system, the number of character lines is substantially larger than on Windows 98, Windows Me, and Windows 95 systems.
Fortran Console applications automatically provide a console. Fortran QuickWin (and Fortran Standard Graphics) applications do not provide a console, but display output and accept input from Fortran statements by using the program window. Different Visual Fortran project types do or do not provide an application console window: Project Type Fortran Console Description of Console Provided Provides a console window since it is intended to be used for character-cell applications that use only text. When you run a Fortran Console application from a command prompt, the existing console environment is used. When you run the application from Windows or Developer Studio (by clicking on ! Execute program-name in the Build menu), a new console environment is created. The limits do not differ greatly whether the console is an existing command prompt or created by Windows or Developer Studio, but Windows NT and Windows 2000 systems have substantially greater capacity than Windows 98, Windows Me, and Windows 95 systems. Basic use of a console is described in Code Samples of Console Use. Fortran QuickWin or Fortran Standard Graphics Does not provide a console, but output to unit 6 and input to unit 5 are directed to the application program window, which can handle both text and graphics. However, because the program window must handle both text and graphics, it is not very efficient for just text-only use. A Fortran QuickWin or Fortran Standard Graphics program window (or child window) provides a console-like window. See Console Use for Fortran QuickWin and Fortran Standard Graphics Applications.
Page 21 of 28
Does not provide a console window, but the user can create a console by using Win32 routines. See Console Use for Fortran Windows Applications and Fortran DLL Applications. Does not provide a console window, but the user can create a console by using Win32 routines. See Console Use for Fortran Windows Applications and Fortran DLL Applications. Depends upon the project type of the main application that references the object code in the library (see above project types).
In addition to the Win32 routines mentioned below, there are other Win32 routines related to console use in the Platform SDK documentation, under Windows Base Services, File and I/O, Consoles and Character-Mode Support. Console Use for Fortran QuickWin and Fortran Standard Graphics Applications For a Fortran QuickWin or Fortran Standard Graphics application, because the default program window handles both graphics and text, the use of a QuickWin window may not be very efficient:
l
QuickWin windows use lots of memory and thus cannot be larger than a certain size. They can be slow to scroll.
Although you can access the console window using WRITE and READ (or other) statements, applications that require display of substantial lines of text, consider creating a DLL that creates a separate console window for efficiency. The DLL application needs to call Win32 routines to allocate the console, display text, accept keyed input, and free the console resources. For a Fortran QuickWin project that uses a DLL to create and use a console, see the CONAPP Visual Fortran sample in ...\Df98\Samples\QuickWin. Basic use of a console is described in Code Samples of Console Use. Console Use for Fortran Windows Applications and Fortran DLL Applications With a Fortran Windows or Fortran DLL application, attempting to write to the console using WRITE and READ (or other) statements before a console is created results in a run-time error (such as error performing WRITE). A console created by a Fortran DLL is distinct from any application console
Page 22 of 28
window associated with the main application. A Fortran DLL application has neither a console nor an application window created for it, so it must create (allocate) its own console using Win32 routines. When used with a Fortran QuickWin or Fortran Standard Graphics application main program, the Fortran DLL can provide its main application with a very efficient console window for text-only use. Like a Fortran DLL application, a Fortran Windows application has neither a console nor an application window created for it, so it must create its own console using Win32 routines. After allocating a console in a Fortran DLL, the handle identifier returned by the GetStdHandle Win32 routine refers to the actual console the DLL creates. When the Fortran Windows application does create a console window, it is very efficient for text-only use. The handle identifier returned by the GetStdHandle Win32 routine refers to the actual console the Fortran Windows application creates. For information about creating a console, see Allocating and Deallocating a Console below. Code Samples for Console Use The following sections shows sample code for using a console:
l
Allocating and Deallocating a Console for Fortran Windows and DLL Applications Extending Size of the Console Window and Console Buffer for console use in any project type Writing and Reading Characters at a Cursor Position for console use in any project type
Allocating and Deallocating a Console To create a console, you use the AllocConsole routine. When you are done with the console, free its resources with a FreeConsole routine. For example, the following code allocates the console, enlarges the buffer size, writes to the screen, waits for any key to be pressed, and deallocates the console:
! The following USE statement provides Fortran interfaces to Win32 routines USE DFWIN ! Begin data declarations integer lines,length logical status integer handle Type(T_COORD) wpos ! Set buffer size variables length = 80 lines = 90
Page 23 of 28
! Begin executable code ! Allocate a console status = AllocConsole() ! get a console window of the currently set size handle = GetStdHandle(STD_OUTPUT_HANDLE) wpos.x = length ! must be >= currently set console window line length wpos.y = lines ! must be >= currently set console window number of lines ! Set a console buffer bigger than the console window. This provides ! scroll bars on the console window to scroll through the console buffer status = SetConsoleScreenBufferSize(handle, wpos) ! Write to the screen as needed. Add a READ to pause before deallocation write (*,*) "This is the console output! It might display instructions or data " write (*,*) " " write (*,*) "Press any key when done viewing " read (*,*) ! Deallocate the console to free its resources. status = FreeConsole()
Calling Win32 routines is described in Calling Win32 Routines and the Visual Fortran Windows Module. If you are using a DLL, your DLL code will need to create subprograms and export their symbols to the main program (see Creating Fortran DLLs). Basic use of a console is described in Extending Size of the Console Window and Console Buffer and Writing and Reading Characters at a Cursor Position. Extending the Size of the Console Window and Console Buffer When you execute a Fortran Console application, the console is already allocated. You can specify the size of the console window, size of the console buffer, and the location of the cursor. If needed, you can extend the size of the console buffer and console window by using the following Win32 routines: 1. You first need to obtain the handle of the console window using the GetStdHandle routine. For example:
! USE statements to include routine interfaces use dflib use dfwin ! Data declarations integer fhandle logical lstat ! Executable code fhandle = GetStdHandle(STD_OUTPUT_HANDLE) ! ...
Page 24 of 28
2. If needed, you can obtain the size of the: Console window by using the GetConsoleWindowInfo routine Console buffer by using the GetConsoleScreenBufferInfo routine For example:
! USE statements to include routine interfaces use dflib use dfwin ! Data declarations integer fhandle logical lstat Type(T_CONSOLE_SCREEN_BUFFER_INFO) conbuf type (T_COORD) dwSize type (T_SMALL_RECT) srWindow fhandle = GetStdHandle(STD_OUTPUT_HANDLE) ! Executable code to get console buffer size lstat = GetConsoleScreenBufferInfo(fhandle, conbuf) write (*,*) " " write (*,*) "Window coordinates= ", conbuf.srWindow write (*,*) "Buffer size= ", conbuf.dwSize ! ...
3. To set the size of the console window and buffer, use the SetConsoleWindowInfo and SetConsoleScreenBufferSize routines with the fhandle value returned previously:
! USE statements to include routine interfaces use dflib use dfwin ! Data declarations integer nlines, ncols logical lstat Type(T_COORD) wpos Type(T_SMALL_RECT) sr Type(T_CONSOLE_SCREEN_BUFFER_INFO) cinfo ! Executable code to set console window size sr.top = 0 sr.left = 0 sr.bottom = 40 ! <= console buffer height -1 sr.right = 60 ! <= console buffer width -1 lstat = SetConsoleWindowInfo(fhandle, .TRUE., sr) ! Executable code to set console buffer size nlines = 100 ncols = 80 wpos.x = ncols ! columns >= console window width wpos.y = nlines ! lines >= console window height lstat = SetConsoleScreenBufferSize(fhandle, wpos) ! ...
Page 25 of 28
You can position the cursor as needed using the SetConsoleCursorPosition routine before you write characters to the screen:
! Use previous data declarations ! Position and write two lines wpos.x = 5 ! 6 characters from left wpos.y = 5 ! 6 lines down lstat = SetConsoleCursorPosition(fhandle, wpos) write(*,*) 'Six across Six down' ! ...
You read from the screen at an appropriate place, but usually you should set the cursor relative to the starting screen location:
! Use previous and the following data declaration CHARACTER(Len=50) charin ! Go back to beginning position of screen wpos.x = 0 ! 0 characters from left wpos.y = 0 ! 0 lines down lstat = SetConsoleCursorPosition(fhandle, wpos) ! Position character input at start of line 11 wpos.x = 0 ! first character from left wpos.y = 10 ! 11 lines down lstat = SetConsoleCursorPosition(fhandle, wpos) read(*,*) charin ! ...
For console I/O, you can use Win32 routines WriteConsoleLine and ReadConsoleLine instead of Fortran WRITE and READ statements.
Read and write arbitrary data from/to the port using SPORT_READ_DATA and SPORT_WRITE_DATA. Read and writes line-terminated data using SPORT_READ_LINE and SPORT_WRITE_LINE.
Once any I/O operation has been requested on the port, an additional thread is started that keeps a read outstanding on the port so that data will not be missed. The SPORT_SET_STATE and SPORT_SET_TIMEOUTS routines allow you to set
Page 26 of 28
basic port parameters such as baud rate, stop bits, timeouts, and so on. Additionally, you can call SPORT_GET_HANDLE to return the Win32 handle to the port so that you can call Win32 Communication Functions to implement additional needs.
An error status of 0 (zero) indicates success Other values are Win32 error values that indicate an error
As described in the calling syntax, these routines require the following USE statement:
USE DFLIB
The USE DFLIB statement includes the routine definitions in the compilation. You may also need to add a USE DWINTY statement to your program because some Win32 constants may be required that are typically defined in the DFWINTY module. Many arguments are optional. If a constant is used where an argument is both input and output, a probe for writeability is done before the output. Thus, in many cases, a constant may be used without creating a temporary variable. It should be noted, however, that doing so may not be compatible with all Fortran implementations.
Page 27 of 28
support that will help the user implement simple applications as well as providing a foundation that can be used for more complex applications. Users needing to implement full serial port protocols (such as a PPP/SLIP implementation or some other complex protocol) should use the Win32 Communication Functions directly to achieve the detailed level of control needed in those cases. Simple tasks, such as communicating with a terminal or some other data collection device are well suited for implementations using the SPORT_xxx routines. You should first familiarize yourself with the hardware connection to the serial device. Typical connections used today involve either a 9 pin/wire connector or a 25 pin/wire connector. Many cables do not implement all 9 or 25 connections in order to save on costs. For certain applications these subset cables may work just fine but others may require the full 9 or 25 connections. All cables will implement the Receive/SendData signals as well as the SignalGround. Without these signals, there can be no data transfer. There are two other categories of important signals:
l
Signals used for flow control Flow control signals tell the device/computer on the other end of the cable that data may be sent or that they should wait. Typically, the RequestToSend/ClearToSend signals are used for this purpose. Other signals such as DataSetReady or DataTerminalReady may also be used. Make sure that the cable used implements all the signals required by your hardware/software solution. Special characters (normally as XON/XOFF) may also be used to control the flow of data instead of or in addition to the hardware signals. Check your specific application to see what cabling is needed.
Signals that indicate status or state of a modem or phone connection. These signals may not be required if the connection between the computer and the device is direct and not through a modem. This signals typically convey information such as the state of the carrier (CarrierDetect) or if the phone line is ringing (Ring). Again, make sure the the cable used implements all the signals required for your application.
After the correct physical connection has been set up the programmer must become familiar with the data protocol used to communicate with the remote device/system. Many simple devices terminate parcels of data with a "record terminator" (often a carriage return or line feed character). Other devices may simply send data in fixed length packets or packets containing some sort of count information. The two types of I/O routines provided by the SPORT_xxx support (line oriented using SPORT_READ_LINE and SPORT_WRITE_LINE or transfer raw data using
Page 28 of 28
SPORT_READ_DATA and SPORT_WRITE_DATA) can handle these two types of data transfer. The programmer must become familiar with the particular application to determine which type of I/O is more appropriate to use.
SPORT_CANCEL_IO SPORT_CONNECT SPORT_GET_HANDLE SPORT_GET_TIMEOUTS SPORT_GET_STATE SPORT_PEEK_DATA SPORT_PEEK_LINE SPORT_PURGE SPORT_READ_DATA SPORT_READ_LINE SPORT_RELEASE SPORT_SET_STATE SPORT_SET_TIMEOUTS SPORT_SHOW_STATE SPORT_SPECIAL_FUNC SPORT_WRITE_DATA SPORT_WRITE_LINE
The SPORT_xxx routines call Win32 routines. For example, the SPORT_SET_STATE routine calls the Win32 routine SetCommState, which uses the DCB Communications Structure. The Win32 Communication Functions and Communication Structures are described in the Platform SDK section on Windows Base Services, subsection Files and I/O, under Communication Reference. For more information:
l
About calling Win32 routines from Fortran programs, see Calling Win32 Routines. About communications and using the Win32 Communications routines, see the Platform SDK descriptions below the following headings: Communications and Communications Functions
Page 1 of 16
Overview of COM and Automation Objects The Role of the Module Wizard Using the Module Wizard to Generate Code Calling the Routines Generated by the Module Wizard Getting a Pointer to an Object's Interface Additional Resources about COM and Automation
For information on creating a COM server with Visual Fortran, see Creating a COM Server.
COM Objects The Component Object Model (COM) provides mechanisms for creating reusable software components. COM is an object-based programming model designed to promote software interoperability; that is, to allow two or more applications or components to easily cooperate with one another, even if they were written by different vendors at different times, in different programming languages, or if they are running on different machines running different operating systems. With COM, components interact with each other and with the system through collections of function calls, also known as methods or member functions or requests, called interfaces. An interface is a semantically related set of member functions. The interface as a whole represents a feature of an object. The member functions of an interface represent the operations that make up the feature. In general, an object can support multiple interfaces and you can use
Using COM and Automation Objects COMQueryInterface to get a pointer to any of them.
Page 2 of 16
The Visual Fortran COM routines provide a Fortran interface to basic COM functions. Automation Objects The capabilities of an Automation object resemble those of a COM object. An Automation object is in fact a COM object that implements the interface IDispatch. An Automation object exposes:
l
Methods, which are functions that perform an action on an object. These are very similar to the member functions of COM objects. Properties, which hold information about the state of an object. A property can be represented by a pair of methods; one for getting the property's current value, and one for setting the property's value.
The Visual Fortran AUTO routines provide a Fortran interface to invoking an automation object's methods and setting and getting its properties. Object Servers COM and Automation objects are made available to users by COM and Automation servers. A COM or Automation server can be implemented either as:
l l
A DLL that is loaded into your process A separate executable program. The separate executable program can reside on the same system as your application or on a different system.
By other programs you install. By creating the object server yourself, for example, by using the Visual Fortran COM Server wizard, Visual C++, or Visual Basic.
For example, the Microsoft visual development environment registers certain objects during installation (see the documentation on the Developer Studio object model). Creating an object server involves deciding what type of object and what type of interfaces or methods should be available. The object's server must
Page 3 of 16
be designed, coded, and tested like any other application. For information about object server creation, see Creating a COM Server. 2. Determine:
Whether the object has a COM interface, Automation interface, or both. Where the object's type information is located.
You should be able to obtain this information from the object's documentation. You can use the OLE-COM Object Viewer tool (provided in the Compaq Visual Fortran program folder) to determine the characteristics of an object on your system. 3. Use the Visual Fortran module wizard to generate code. The Visual Fortran module wizard is a application that interactively asks certain questions about the object, including its name, type, and other information. The information collected by the module wizard is used in the generated code. To learn about using the Visual Fortran module wizard, see Using the Module Wizard to Generate Code. 4. Write a Fortran 90 program to invoke the code generated by the Visual Fortran module wizard. To understand more about calling the interfaces and jacket routines created by the module wizard, see Calling Routines Generated by the Module Wizard.
Page 4 of 16
routines in a DLL. You can store a type library in a file of its own (usually with an extension of .TLB) or it can be part of another file. For example, the type library that describes a DLL can be stored in the DLL itself. After you start the Module Wizard (Tools menu, Fortran Module Wizard), a dialog box requests the source of the type information that describes the object you need to use. You need to determine what type of object it is (or DLL) and how it makes its type information available. The choices are:
l l l l l
Automation Object Type Library containing Automation information Type Library containing COM interface information Type Library containing DLL information DLL containing type information
Object servers typically contain the type library in the same file (either .dll or .exe) as the object implementation. Many objects implement dual interfaces, which supports both COM and Automation. For an object that supports dual interfaces, you can choose either of these options:
l l
Type Library containing Automation information Type Library containing COM interface information
The COM object interfaces tend to be more efficient (better run-time performance). ActiveX controls implement an Automation interface. When using an ActiveX control, choose the Type Library containing Automation interface information option. Most DLLs do not provide a type library that describes the programming interfaces in the DLL, and therefore cannot be used by the Module Wizard. For these reasons, the Automation Information (second option) or COM Interface Information (third option) are the most commonly used. The following initial screen appears after you select the Visual Fortran Module Wizard: Initial Module Wizard Screen
Page 5 of 16
After you select one of the five choices, one of two different screens will appear depending on the selection made. The "Module Name" in the initial Module Wizard screen is used as the name of the Fortran module being generated. It is also used as the default file name of the generated source file. If You Select Automation Object If you select Automation Object, the following screen appears: Application Object Screen
Page 6 of 16
Microsoft recommends that object servers provide a type library. However some applications do not, but do provide type information dynamically when running. Use this option for such an application. Enter the name of the application, name of the object, and version number. The version number is optional. If you do not specify it, you will get the latest version of the object. Note that this method only works for objects that provide a programmatic identifier (ProgID). ProgIDs are entered into the system registry and identify, among other things, the executable program that is the object's server. After entering the information and pressing the Generate button, the Fortran Module Wizard asks you for the name of the source file to be generated. It then asks COM to create an instance of the object identified by the ProgID that the wizard constructs using the supplied information. COM starts the object's server if it needs to do so. The wizard then asks the object for its type information and generates a file containing Fortran modules. If You Select Other Options After selecting any of the remaining options in the initial screen and press the "Next" button, the Module Wizard displays the following screen: Type Library Screen
Page 7 of 16
Choose the type library (or file containing the type library), and optionally specific components of the type library. At the top of the dialog box is a combo box that lists all of the type libraries that have been registered with the system. You will notice a number of different file extensions, for example, .OLB (object libraries) and .OCX (ActiveX controls). Select a type library from the list or press Browse to find the file using the standard "Open" dialog box. Once you have selected a type library press the Show button to list the components you can select from the type library. By default, the Fortran Module Wizard will use all of the components. Optionally, you can select the ones desired from the list. After entering the information and pressing the "Generate" button, the Fortran Module Wizard asks you for the name of the source file to be generated. It then asks COM to open the type library and generates a file containing Fortran modules. The Fortran Module Wizard also has a command-line interface. The MODWIZ command has the following form: MODWIZ [options] typeinfo-name For a list of MODWIZ command options and an explanation of typeinfo-name, type the following command in a Fortran Command Prompt (available from the Visual Fortran program folder):
MODWIZ /?
Derived-type definitions are Fortran equivalents of data structures that are found in the type information. Constant definitions are Fortran PARAMETER declarations that include
Page 8 of 16
identifiers and enumerations found in the type information. Procedure interface definitions are Fortran interface blocks that describe the procedures found in the type information. Procedure definitions are Fortran functions and subroutines that are jacket routines for the procedures found in the type information.
The jacket routines make the external procedures easier to call from Fortran by handling data conversion and low-level invocation details. The use of modules allows the Visual Fortran Module Wizard to encapsulate the data structures and procedures exposed by an object or DLL in a single place. You can then share these definitions in multiple Fortran programs. The appropriate USE statement needs to be added in your program, as well as function invocations or subroutine calls. The routines generated by the Visual Fortran Module Wizard are designed to be called from Fortran. These routines in turn call the appropriate system routines (not designed to be called from Fortran), thereby simplifying the coding needed to use COM and Automation objects. Visual Fortran provides a set of run-time routines that present to the Fortran programmer a higher level abstraction of the COM and Automation functionality. The Fortran interfaces that the Wizard generates hide most of the differences between Automation objects and COM objects. Depending on the options specified, the following routines can be present in the generated code: DFCOM Routines (COMxxxxx) COMAddObject Reference COMCLSIDFromProgID COMCLSIDFromString COMCreateObjectByGUID Adds a reference to an object's interface. Passes a programmatic identifier and returns the corresponding class identifier. Passes a class identifier string and returns the corresponding class identifier. Passes a class identifier and creates an instance of an object. It returns a pointer to the object's interface. Passes a programmatic identifier and creates an instance of an object. It returns a pointer to the object's IDispatch interface. Pass a class identifier and returns a pointer to the interface of a currently active object.
COMCreateObjectByProgID COMGetActiveObjectByGUID
Page 9 of 16
Passes a programmatic identifier and returns a COMGetActiveObjectByProgID pointer to the IDispatch interface of a currently active object. COMInitialize COMIsEqualGUID COMGetFileObject COMQueryInterface COMReleaseObject COMStringFromGUID COMUninitialize Initializes the COM library. You must initialize the library before calling any other COM or AUTO routine. Determines if two GUIDs are the same. Passes a file name and returns a pointer to the IDispatch interface of an Automation object that can manipulate the file. Passes an interface identifier and it returns a pointer to an object's interface. Indicates that the program is done with a reference to an object's interface. Passes a GUID and returns the corresponding string representation. Uninitializes the COM library. This must be the last COM routine that you call. Passes an argument name and value and adds the argument to the argument list data structure. Allocates an argument list data structure that holds the arguments that you will pass to AUTOInvoke. Deallocates an argument list data structure. Retrieves the exception information when a method has returned an exception status. Passes the name or identifier of the property and gets the value of the Automation object's property. Passes the member ID of the property and gets the value of the Automation object's property into the argument list's first argument. Passes an argument list data structure and gets the value of the Automation object's property specified in the argument list's first argument. Passes the name or identifier of an object's method and an argument list data structure. It invokes the method with the passed arguments.
AUTOGetPropertyByID
AUTOGetPropertyInvokeArgs
AUTOInvoke
Page 10 of 16
AUTOSetProperty
Passes the name or identifier of the property and a value. It sets the value of the Automation object's property. Passes the member ID of the property and sets the value of the Automation object's property using the argument list's first argument. Passes an argument list data structure and sets the value of the Automation object's property specified in the argument list's first argument.
AUTOSetPropertyByID
AUTOSetPropertyInvokeArgs
Visual Fortran Samples include several projects in the ...\DF98 \SAMPLES\ADVANCED\COM folder that demonstrate the use of the Fortran Module Wizard. For example:
l
uses Automation to drive Microsoft Excel 97 to create a chart from Fortran data. DSBUILD uses Automation to drive the visual development environment to rebuild a project configuration. DSLINES uses COM to drive the Microsoft visual development environment to edit a Fortran source file and convert Debug lines (column 1) to IFDEF directives. IWEB uses COM interfaces to start a Web browser and direct the browser to open a specified URL.
AUTODICE
Example of Generated Code Used by the DSLINES Sample The DLINES Sample contains the code that invokes this and other Microsoft visual development environment functionality using COM interfaces. The following code shows an annotated version of a portion of the code generated by the Fortran Module Wizard from the COM type information in ...\Common\MSDev98\Bin\devshl.dll. This COM type information describes the top-level objects in the Microsoft visual development environment object model.
INTERFACE ! Saves the document to disk. INTEGER*4 FUNCTION IGenericDocument_Save($OBJECT, vFilename, & vBoolPrompt, pSaved) USE DFWINTY INTEGER*4, INTENT(IN) :: $OBJECT ! Object Pointer !DEC$ ATTRIBUTES VALUE :: $OBJECT TYPE (VARIANT), INTENT(IN) !DEC$ ATTRIBUTES VALUE TYPE (VARIANT), INTENT(IN) !DEC$ ATTRIBUTES VALUE :: :: :: :: vFilename ! (Optional Arg) vFilename vBoolPrompt ! (Optional Arg) vBoolPrompt
Page 11 of 16
Notes for this example: If the type information provides a comment that describes the member function, then the comment is placed before the beginning of the procedure. The first argument to the procedure is always $OBJECT. It is a pointer to the object's interface. The remaining argument names are determined from the type information. For information on how to get a pointer to an object's interface, see Getting a Pointer to an Object's Interface. This is an example of an ATTRIBUTES directive statement used to specify the calling convention of an argument. A VARIANT is a data structure that can contain any type of Automation data. It contains a field that identifies the type of data and a union that holds the data value. The use of a VARIANT argument allows the caller to use any data type that can be converted into the data type expected by the member function. Nearly every COM member function returns a status of type HRESULT. Because of this, if a COM member function produces output it uses output arguments to return the values. In this example, the "pSaved" argument returns a routine specific status value. The interface of a COM member function looks very similar to the interface for a dynamic link library function with one major exception. Unlike a DLL function, the address of a COM member function is never known at program link time. You must get a pointer to an object's interface at run-time, and the address of a particular member function is computed from that.
The following code shows an annotated version of the wrapper generated by the Fortran Module Wizard for the "Save" function. The name of a wrapper is the same as the name of the corresponding member function, prefixed with a "$" character.
! Saves the document to disk. INTEGER*4 FUNCTION $IGenericDocument_Save($OBJECT, vFilename, & vBoolPrompt, pSaved) !DEC$ ATTRIBUTES DLLEXPORT :: $IGenericDocument_Save IMPLICIT NONE
Page 12 of 16
INTEGER*4, INTENT(IN) :: $OBJECT ! Object Pointer !DEC$ ATTRIBUTES VALUE :: $OBJECT TYPE (VARIANT), INTENT(IN), OPTIONAL :: vFilename !DEC$ ATTRIBUTES REFERENCE :: vFilename TYPE (VARIANT), INTENT(IN), OPTIONAL :: vBoolPrompt !DEC$ ATTRIBUTES REFERENCE :: vBoolPrompt INTEGER*4, INTENT(OUT) :: pSaved ! Void !DEC$ ATTRIBUTES REFERENCE :: pSaved INTEGER*4 $RETURN INTEGER*4 $VTBL ! Interface Function Table POINTER($VPTR, $VTBL) TYPE (VARIANT) :: $VAR_vFilename TYPE (VARIANT) :: $VAR_vBoolPrompt IF (PRESENT(vFilename)) THEN $VAR_vFilename = vFilename ELSE $VAR_vFilename = OPTIONAL_VARIANT END IF IF (PRESENT(vBoolPrompt)) THEN $VAR_vBoolPrompt = vBoolPrompt ELSE $VAR_vBoolPrompt = OPTIONAL_VARIANT END IF $VPTR = $OBJECT ! Interface Function Table $VPTR = $VTBL + 84 ! Add routine table offset IGenericDocument_Save_PTR = $VTBL $RETURN = IGenericDocument_Save($OBJECT, $VAR_vFilename, & $VAR_vBoolPrompt, pSaved) $IGenericDocument_Save = $RETURN END FUNCTION $IGenericDocument_Save
Notes for this example: The wrapper takes the same argument names as the member function interface. The wrapper computes the address of the member function from the interface pointer and an offset found in the interface's type information. In implementation terms, an interface pointer is a pointer to a pointer to an array of function pointers called an "Interface Function Table". Arguments to a COM or Automation routine can be optional. The wrapper handles the invocation details for specifying an optional argument that is not present in the call. The offset of the "Save" member function is 84. The code assigns the computed address to the function pointer IGenericDocument_Save_PTR, which was declared with the interface shown above, and then calls the function.
The DLINES Sample contains the code that invokes this and other Microsoft visual development environment functionality using COM interfaces.
Page 13 of 16
Example of Generated Code Used by the DSBUILD Sample The DSBUILD example contains the code that invokes this and other Microsoft visual development environment functionality using Automation interfaces. The following code shows an annotated version of a portion of the code generated by the Fortran Module Wizard from the Automation type information in ...\Common\MSDev98\Bin\devshl.dll.
! Rebuilds all files in a specified configuration. SUBROUTINE IApplication_RebuildAll($OBJECT, Configuration, $STATUS) !DEC$ ATTRIBUTES DLLEXPORT :: IApplication_RebuildAll IMPLICIT NONE INTEGER*4, INTENT(IN) :: $OBJECT ! Object Pointer !DEC$ ATTRIBUTES VALUE :: $OBJECT TYPE (VARIANT), INTENT(IN), OPTIONAL :: Configuration !DEC$ ATTRIBUTES REFERENCE :: Configuration INTEGER*4, INTENT(OUT), OPTIONAL :: $STATUS ! Method status !DEC$ ATTRIBUTES REFERENCE :: $STATUS INTEGER*4 $$STATUS INTEGER*4 invokeargs invokeargs = AUTOALLOCATEINVOKEARGS() IF (PRESENT(Configuration)) CALL AUTOADDARG(invokeargs, '$ARG1', & Configuration, AUTO_ARG_IN) $$STATUS = AUTOINVOKE($OBJECT, 28, invokeargs) IF (PRESENT($STATUS)) $STATUS = $$STATUS CALL AUTODEALLOCATEINVOKEARGS (invokeargs) END SUBROUTINE IApplication_RebuildAll
Notes for this example: The first argument to the procedure is always $OBJECT. It is a pointer to an Automation object's IDispatch interface. The last argument to the procedure is always $STATUS. It is an optional argument that you can specify if you wish to examine the return status of the method. The IDispatch Invoke member function returns a status of type HRESULT. An HRESULT is a 32-bit value. It has the same structure as a Win32 error code. In between the $OBJECT and $STATUS arguments are the method arguments' names determined from the type information. Sometimes, the type information does not provide a name for an argument. The Fortran Module Wizard creates a "$ARGn" name in this case. AUTOAllocateInvokeArgs allocates a data structure that is used to collect the arguments that you will pass to the method. AUTOAddArg adds an argument to this data structure. AUTOInvoke invokes the named method passing the argument list. This returns a status result.
Page 14 of 16
If the caller supplied a status argument, the code copies the status result to it. AUTODeallocateInvokeArgs deallocates the memory used by the argument list data structure.
The DSBUILD Sample in the ...\DF98\Samples\Advanced\COM\ folder contains the code that invokes this and other Microsoft visual development environment functionality using Automation interfaces.
Obtaining the Pointer to an Object's Interface To use the routines generated by the Module Wizard, your application must get a pointer to an object's interface. This pointer is used as the value of the $OBJECT argument, which is the first argument of every interface generated by the Module Wizard. Typically, your application obtains its first object pointer by calling the COM routine COMCreateObject. COMCreateObject creates a new instance of an object class. COMCreateObject is the generic name of the two subroutines COMCreateObjectByProgID and COMCreateObjectByGUID:
l
Use COMCreateObjectByProgID to create Automation objects. It accepts the progID of an object and returns a pointer to the object's IDispatch interface. Use COMCreateObjectByGUID to create both COM and Automation objects.
Page 15 of 16
The first argument is a class identifier (CLSID) that uniquely identifies the object's class. The Module Wizard defines a GUID parameter for each class selected from the Type library. These are given the name in the form: CLSID_class-name. The second argument allows you to limit the type(s) of server that the call will accept. Most of the time you can use CLSCTX_ALL to accept any type of server. The third argument is an interface identifier (IID) that specifies the particular object interface you are requesting: To request an Automation interface, use IID_IDispatch. To request a COM interface, the Module Wizard defines a GUID parameter for each interface selected from the type library. These are given a name in the form: IID_interface-name. The fourth argument is an output parameter that returns the object's interface pointer.
The COMCreateObjectByProgID and COMCreateObjectByGUID subroutines return an interface pointer of an object that the server has defined as being externally creatable. However, not all objects are externally creatable. Often, a server implements a hierarchy of objects, or object model. COMCreateObject is called to obtain a pointer to an interface of the root object in the hierarchy. Methods and/or properties of the root object are used to obtain pointers to child objects, and so on, down the hierarchy. You can see examples of this in the Autodice, Dlines, and DSbuild Visual Fortran samples. All objects must implement the IUnknown interface. Every object also implements one or more additional interfaces. You can always get a pointer to any of the object's interfaces from any of the object's interface pointers by using the COMQueryInterface subroutine. It is important that you have a pointer to the correct object interface when calling a routine generated by the Module Wizard. If not, your application will likely crash. Releasing the Pointer to an Object's Interface When you have finished using an object's interface pointer, you must call COMReleaseObject with the pointer. This includes object pointers that you have received using any method, including COMCreateObject, COMQueryInterface, or by calling an object's method.
Page 16 of 16
Compaq lists some of these publications as additional resources to assist customers who want to learn more about the subject matter. This list does not commenteither negatively or positivelyon any documents listed or not yet listed. Books and related resources about COM and Automation include:
l
Inside COM+ Base Services by Guy Eddon, Henry Eddon. Published by Microsoft Press (Redmond, Washington) 1999 Understanding COM+ by David S. Platt. Published by Microsoft Press (Redmond, Washington) 1999 Inside Distributed COM by Guy Eddon; Henry Eddon. Published by Microsoft Press (Redmond, Washington) 1998 Inside COM by Dale Rogerson. Published by Microsoft Press (Redmond, Washington) 1996 Understanding ActiveX and OLE by David Chappell. Published by Microsoft Press (Redmond, Washington) 1996 Automation Programmer's Reference by Microsoft. Published by Microsoft Press (Redmond, Washington) 1997 ActiveX Controls Inside Out, Second Edition by Adam Denning. Published by Microsoft Press (Redmond, Washington) 1997 Platform SDK online version (HTML Help Viewer, provided with MSDN and Visual Fortran). Relevant titles include Platform SDK, COM and ActiveX Object Services. Visual C++ User's Guide online version (HTML Help Viewer, provided with MSDN and Visual Fortran)
Page 1 of 41
Advantages of a COM Server What You Need to Provide and What the Fortran COM Server Wizard Will Do Using the Fortran COM Server Wizard, which includes: Specifying the Fortran COM Server as a Developer Studio Add-in Creating a Fortran COM Server Project Using the Fortran COM Server Wizard to Define your COM Server Adding a Property Working with the Hierarchy Pane Description of Property Pages Modifying the Generated Code Build Notes Visual C++ ClassView Using Example Clients Modifying Your COM Servers Interface Design Considerations, which includes: Method and Properties Data Types COM Status Codes: HRESULT Visual Basic and Visual C++ Client Notes Advanced COM Server Topics, which includes: Choosing Between DLL or EXE COM Servers DLL Server Surrogates Discussion of Wizard Code Generation Adding Support for COM ErrorInfo Objects Threading Models Marshalling, Proxies and Stubs A Map of the Generated "Do Not Edit" Code Deploying the Server on Another System
Page 2 of 41
A COM server is a reusable component, which allows multiple applications to use the server. The classes specified by the server define a "contract" between the server and its clients. The server can change the specific implementation of the functionality without breaking the contract. That is, without requiring clients to be changed or rebuilt. A COM server is programming-language independent. Multiple development tools can be used to access the server's functionality, including Visual Basic, Visual C++, and Visual Fortran. A COM server is self-describing. The server provides a type library that describes the classes and interfaces. Many tools can take advantage of this information and relieve the client programmer from needing to understand low-level invocation details, such as calling conventions. This is a great improvement over multi-language programming with DLLs, where the client programmer has to understand the details of data types and calling conventions. A COM server is self-registering. The clients do not need to worry about where the server is located on their system, as COM finds this information in the system registry. A COM server can be implemented as an in-process server. Like a DLL, it is loaded into the client's process. A COM server can also be implemented as a separate application and can even reside on a separate machine.
Visual Fortran provides the Fortran COM Server Wizard. The wizard generates the Fortran code necessary to implement a COM server, and allows you to concentrate on the code that is specific to the functionality that your server provides to its clients. As explained in Overview of COM and Automation Objects, COM supports two types of servers: COM servers and Automation servers. The Fortran COM Server Wizard can only create a COM server or a server that supports dual interfaces. The wizard cannot create an Automation-only interface. For more information about creating COM servers using Visual Fortran, see What You Need to Do and What the COM Server Wizard Will Do.
Page 3 of 41
What You Need to Provide and What the Fortran COM Server Wizard Will Do
The Fortran COM Server Wizard generates a Developer Studio project and a number of source files that implement all of the infrastructure of a COM server. You use the Fortran COM Server Wizard's user interface to define the implementation of one or more object classes, including its interface(s), and method(s). You need to define fields in the derived-type for each instance and to write code for those fields, if necessary. You also need to write code needed to implement each method. What Information You Need to Provide To use the Fortran COM Server Wizard to create a COM server, you first need to describe to the Fortran COM Server Wizard the class(es) that you want to implement. A class implements one or more COM interfaces. In COM terminology, an interface is a semantically related set of functions. The interface as a whole represents a "feature", or set of related functionality, that is implemented by the class. An interface contains methods, otherwise known as member functions. A method is a routine that performs one of the actions that make up the feature. When using the Fortran COM Server Wizard, methods are Fortran functions that take arguments and return a value like any other Fortran function. Consider a simple example of a class that we will create using the Fortran COM Server Wizard, called AddingMachine. The class contains a single interface that we call IAdd. By convention, all interface names begin with a capital letter "I". We define three methods in the IAdd interface:
l
Clear, which takes no arguments and sets the current value of the adding machine to 0. Add, which takes a single REAL argument, the amount to add to the current value. GetValue, which returns the current value.
The Clear method, Add method, and GetValue method allow you to perform specific, distinct tasks with the IAdd interface from any language that supports a COM client. The Fortran COM Server Wizard provides a user interface to enter this information about the class (in this case the AddingMachine class), which is discussed later in Using the COM Server Wizard. An interface can also contain properties. Properties are method pairs that set or return information about the state of an object. When you add a property to an interface, you are actually adding one or two methods. This is discussed further
Page 4 of 41
Another important concept is the data associated with an object. A key concept of object-oriented programming is encapsulation. Encapsulation means that all of the details about how the object is implemented, including the data that it uses and the logic that it uses to perform its work, is hidden from the client. The client's only access to the object is through the interfaces that the object supports. You need to define the data that the object uses and code the logic of the methods. For the data, the Fortran COM Server Wizard uses the model that each instance of the object has an associated instance of a Fortran derived-type. The code generated by the wizard takes care of creating and destroying the instances of the derived-type as objects are created and destroyed. You define the fields of the derived-type. For example, with our AddingMachine, each AddingMachine object needs to store the current value. The derived-type associated with each AddingMachine object would contain a single field of type REAL. We name it CurrentValue. Note that each instance of the AddingMachine object has its own instance of the derived-type and its own CurrentValue. This means that the server could support multiple clients simultaneously and each client would see its own AddingMachine. That is, each client is unaffected by the existence of other clients. The derived-type associated with each object is discussed in detail in Creating a Fortran COM Server Project. To summarize, at a high level, what you need to do to create a COM server using the Fortran COM Server Wizard:
l
Define the class(es), interface(s), method(s), and properties using the Wizard's user interface. Define the fields in the derived-type that is associated with each instance of a class. Write code that initializes the fields in the derived-type (if necessary), and code that releases any resources used by the fields in the derived-type (if necessary). Write the code that implements the methods.
What the COM Server Will Provide The Fortran COM Server Wizard generates a Developer Studio project and a number of source files that implement all of the infrastructure, or "plumbing", of the COM server. The generated files take care of such tasks as:
l
l l
Defining the GUIDs that uniquely identify your classes and interfaces. For a discussion of GUIDs, see Getting a Pointer to an Object's Interface. Registering the server and your classes and interfaces on the system. Implementing the class factory that creates instances of your object.
Page 5 of 41
Creating the Interface Definition Library (IDL) and type library that describes your classes and interfaces. For a discussion of type libraries, see Using the Module Wizard to Generate Code. Implementing the IUnknown and IDispatch interfaces for your object. All of the interfaces created by the Fortran COM Server Wizard are derived from IUnknown or IDispatch (dual interfaces). Creating and destroying instances of the derived-type associated with your object. Invoking the Fortran routines that implement your object's methods.
The majority of these source files are generated fully by the Fortran COM Server Wizard and are not modified by you. Other files contain the skeleton or template of your derived-type and methods. You edit these Fortran source files to fill in your implementation. You will see how this is done as we work through the AddingMachine example. For information on using the Fortran COM Server Wizard user interface to create a COM server, see Using the COM Server Wizard.
Specifying the Fortran COM Server as a Developer Studio Add-in Creating a Fortran COM Server Project Using the Fortran COM Server Wizard to Define Your COM Server Adding a Property Working with the Hierarchy Pane Description of Property Pages Modifying the Generated Code Build Notes Visual C++ ClassView Using Example Clients Modifying Your COM Servers
Specifying the Fortran COM Server as a Developer Studio Add-in The Fortran COM Server Wizard is implemented as a Developer Studio Add-in. After you install Visual Fortran, you need to register and load the Fortran COM Server Wizard on your system. If you have not already done so, do the following: 1. In the Tools menu, click Customize 2. Click the Add-in and Macros tab
Page 6 of 41
3. Enable the check box for the Fortran COM Server Wizard if it is displayed in the list 4. If it is not displayed, click the Browse button and find the CSAddin.dll file, in the ...\Microsoft Visual Studio\Common\MSDev98\Addins\Df98\ directory. 5. Click the Close button You only need to perform this procedure once on your system. Creating a Fortran COM Server Project The first step in creating a Fortran COM Server is to create a new project. Start Developer Studio. In File menu, click New. In the New Projects dialog box, select (click) the Fortran COM Server project type, as shown below (if Microsoft Visual C++ is installed on your system, additional project types will appear):
The Fortran COM Server Wizard is implemented as a Developer Studio Add-in and must be registered and loaded on your system. If you receive a message that begins with "The Fortran COM Server Wizard Add-in is not loaded into Developer Studio ...", follow the instructions in Initializing the Fortran COM Server as a Developer Studio Add-in. In this case:
Page 7 of 41
1. Enter "Adder" as the name of the project. 2. Accept or modify the project folder location. 3. Click the OK button. (If you click <Back, the previous screen appears, allowing you to change the name, location, or project type of the project being created.) To define the initial attributes of the Fortran COM server project being created, additional information is requested. The following screen shows the Fortran COM Server AppWizard. You use the project AppWizard once per project to create the project files and skeleton template (as with other project types). The Fortran COM Server AppWizard requests the type of server (DLL or EXE), the class name, interface name, and the class derived type name:
In this case: 1. Accept the default server type (DLL). 2. Type the class name AddingMachine. Default text appears for the default interface name and class derived type name. 3. Shorten the default interface name to IAdd. 4. Accept the default class derived type name. 5. Click the Finish button. (If you click Cancel, project creation is terminated.) A summary screen appears that summarizes the location and template information created for this project:
Page 8 of 41
After you click OK, the project is created and the Fortran COM Server Wizard appears. Using the Fortran COM Server Wizard to Define your COM Server The Fortran COM Server Wizard lets you interactively define the attributes of your COM server. It lets you define and name classes, interfaces, methods, and so on. The user interface contains two panes:
l
The left pane is a tree control that displays your classes, interfaces, methods, and so on in a hierarchy. The right pane is a property page that displays the properties of the selected element in the hierarchy.
The Fortran COM Server Wizard is automatically invoked by the Fortran COM Server AppWizard, so you can enter the initial definition of your server. You can, but you are not required to, fully define the server at this point. You can partially define your project and close your project workspace. Later, after you start Developer Studio and open the project workspace again, you can start the Fortran COM Server Wizard by clicking the Fortran COM Server Wizard menu item in the View menu. With our example COM server, the initial screen of the Fortran COM Server
Page 9 of 41
Wizard now appears, displaying the type library GUID and the type library version:
In this case, you need to expand the hierarchy to select the IAdd interface name: 1. Click the plus sign (+) to open the COM server name (and project name) Adder. 2. Click the plus sign (+) to open the class name AddingMachine. 3. Click the IAdd interface name. The screen appears as follows:
Page 10 of 41
If the Dual Interface check box is checked, click the Dual Interface check box to uncheck it. The Adder sample only supports a COM server interface. The hierarchy is as follows:
l l
l l
The root is the COM server itself, Adder. This is also the project name. The immediate children of the COM server are classes. Our hierarchy initially contains a single class, AddingMachine. You can add additional classes to the COM server. The immediate children of a class are interfaces and the class derived-type. Our hierarchy contains the interface IAdd and the class derived-type AddingMachine_InstanceData. Each class contains one, and only one, class derived-type. You can add additional interfaces to the class. The immediate children of an interface are methods (one or more). The immediate children of a method are the method arguments (one or more).
You can use the Fortran COM Server Wizard's graphical user interface to add a class, interface, method, property, or argument: 1. 2. 3. 4. Select the parent item. Click the Add button. Type the requested name (and other requested text). Click OK.
To delete an item: 1. Select the item to be deleted. 2. Click the Delete button. For our example Adder COM server, add the methods to the IAdd interface: 1. Verify that the IAdd interface is selected (highlighted). 2. Click the New button. The following screen appears:
Page 11 of 41
The active buttons displayed depend on the selected (highlighted) item (COM server, class, interface, method, or argument) in the hierarchy. The active buttons represent the types that are valid in the current context. For example, the Argument button is inactive since an interface was selected and an Argument must be added to a method. In this case: 1. Select the Method type. 2. Type Clear as the name of the first method of the IAdd interface. 3. Click OK. Because the Clear method takes no arguments, you can now add a second method, named Add: 1. 2. 3. 4. Click (select) the IAdd interface. Click the New button. Type the second method name, Add. Click OK.
The Add method requires an argument, which we will name Operand. To add the Operand argument: 1. 2. 3. 4. Click (select) the Add method. Click the New button. Type the argument name, Operand. Click OK.
Page 12 of 41
After you click OK, the Fortran COM Server Wizard screen adds the Operand argument and displays its properties. The default data type of an argument is INTEGER(4). In our example: 1. Use the Fortran data type combo box to change the data type to REAL(4) (as follows):
2. Leave the default values for Interface data and Intent. We need to add the last method: 1. Click (select) the IAdd interface.
Creating a COM Server 2. Click the New button. 3. Type the method name, GetValue. 4. Click OK.
Page 13 of 41
The GetValue method requires an argument, which we will name CurrentValue. To add the CurrentValue argument: 1. 2. 3. 4. Select the GetValue method. Click the New button. Type the argument name, CurrentValue. Click OK.
For the CurrentValue argument: 1. 2. 3. 4. Use the Fortran data type combo box to change the data type to REAL(4) Leave the default values for Interface data. Change the Intent to Out. Click the Return value check box.
In our example, the server definition is now complete. Click the Save button to dismiss the Fortran COM Server Wizard. The project is now displayed, including the TODO.TXT file, allowing you to modify the generated code.
Page 14 of 41
For detailed information about the property pages (right pane) for server, class, interface, method, argument, or instance properties, see Description of Property Pages. To change the definition of your server later, select the Fortran COM Server Wizard menu item in the View menu. Adding a Property In addition to methods, an interface can contain properties. Properties are method pairs that set or return information about the state of an object. Typically, a property corresponds to a field in the object's derived type. For example, in the AddingMachine example, we could have used a CurrentValue property rather than a GetValue (and SetValue) methods. A property is primarily an Automation concept. In a COM interface, a property is implemented as one or two methods:
l l
A property can be defined as read-only or write-only. In this case, there will be a single get_ or put_ method. Defining an interface member as a property, rather than a method, can affect the syntax used by the client of your server. For example, Visual Basic uses a different syntax for getting or putting a property than for invoking a method. If the client language has a specific property syntax, it typically looks similar to setting a field in a record, rather than making a call. To add a property to your interface: 1. Click (select) the interface in the hierarchy pane. 2. Click the New button. The New dialog appears. Do the following: Select the Property type. Type a name for the property. The completed screen appears as follows:
Page 15 of 41
3. In the New Property dialog box, define the attributes of the property. This includes: Whether the property has both get_ and put_ methods. The integer Automation identifier of this property. The data type of this property. 4. Click OK. One or two methods are added to your interface. Two methods are added if you select both the get_ and put_ methods. One method is added if it is read-only or write-only. The properties methods and the argument that identifies the property value are fully defined and do not need to be modified. A property method typically only has the single argument that represents the property value. You can add additional arguments if it makes sense. For example, additional arguments might be added if the property is an array of values, where an integer index argument is added to the property method. When adding additional arguments to a property method, make sure that they are added before the property value argument. That is, the property value argument must always be the last argument to the property method. You can use drag-and-drop in the hierarchy pane to change the order of arguments. The order in the method is always the order that appears in the hierarchy pane. Working with the Hierarchy Pane
Page 16 of 41
As you have seen in the AddingMachine example, the hierarchy pane represents the definition of your server; that is, the classes, interfaces, methods, and so on. The hierarchy pane's user interface supports the following functionality: Expand/contract an area When the Fortran COM Server wizard is initially displayed, the hierarchy is fully contracted. Click the plus sign (+) next to an item to display its children. At this point, the + changes to a -. Click the - to hide the children. Doubleclicking on an item also toggles its expand/contract state. To add a new entry to the hierarchy, select the entry that will precede the new entry in the hierarchy. Then click the New... button or click the right mouse button to display the pop-up hierarchy pane context menu and select one of the New... entries in the context menu. To delete an entry from the hierarchy, select the item to be deleted. Then click the Delete button or click the right mouse button to display the pop-up hierarchy pane context menu and select the Delete entry in the context menu. A confirmation dialog box is displayed to ensure that you do not accidentally delete items. All of an item's children are deleted when the item is deleted. To rename an entry in the hierarchy, select the item to be renamed. Wait a few seconds (to avoid double-clicking) and then click the item again. The item's name will enter editing mode allowing you to change the name. Press the Enter key when you have finished editing the name. The order of some of the entries in the hierarchy is very important. In particular:
l
Delete an entry
Rename a member
The order of the methods in an interface define the order of the methods in the class' VTBL. Changing the order of the methods will break an existing client. The order of the arguments in a method define the order of the arguments in the method's interface. Changing the order of the arguments will break an existing client.
The hierarchy pane supports drag-and-drop to allow you to change the order of items. To move an item, click and drag the item and drop it on the item that you want to precede it in the hierarchy.
Page 17 of 41
Description of Property Pages Property pages appear in the right pane of the Fortran COM Server Wizard window. Different property pages apply to:
l l l l l l
Server Properties Class Properties Interface Properties Method Properties Argument Properties Instance Type Properties
Server Properties: Type Library GUID Type Library Version The unique identifier of the server's type library. There is usually no reason to change this from the default value generated by the Fortran COM Server Wizard. The current version of the Type Library.
Class Properties: The version independent program ID (or text alias) for the class. The ProgID can be used in calls such as COMCreateObjectByProgID. The current version of the class. It is appended to the ProgID to define the version-specific ProgID. A short name for the class. It is used in some of the generated file names. A string used as the default value of the class' ProgID keys in the registry. This string is often used by tools, such as the OLECOM Object Viewer, that display a list of the objects that are registered on the system. A string used to set the class' help string attribute in the IDL file.
ProgID
Description
Help String
Page 18 of 41
Threading model
The threading model of the class. The two choices are Apartment and Single. See Threading Models in Advanced COM Server Topics for information about the implications of this choice. The unique identifier of the class. There is usually no reason to change this from the default value generated by the Wizard.
CLSID
Interface Properties: If checked, then the dual interface attribute is set in the IDL file. A dual interface supports both COM and Automation clients. If checked, then the interface uses only Automationcompatible data types as described in Method Data Types. If checked, then the default interface attribute is set for this interface in the IDL file. The default attribute represents the default programmability interface of the object, and is intended for use by macro languages. A string used to set the interface's help string attribute in the IDL file. The unique identifier of the interface. There is usually no reason to change this from the default value generated by the Wizard.
Help String
IID
Method Properties: ID Help string Property Method The identifier of the method used by Automation clients. A string used to set the method's help string attribute in the IDL file. If checked, then the method is the get_ or put_ method of a property.
Argument Properties:
Page 19 of 41
The Fortran data type of the argument. Select one of the data types from the list, or type in the data type. See Method Data Types for a discussion of the implications of your choice. The IDL data type. If you select one of the Fortran data types from the predefined list, then this field defaults to the corresponding IDL data type. Select one of the data types from the list, or type in the data type. See Method Data Types for a discussion of the implications of your choice The INTENT of the argument, one of the following list:
l
Intent
In - the argument value is read but not modified by the method Out - the argument value is not read, but is modified by the method InOut - the argument value is both read and modified by the method
By Reference
Indicates that an argument is passed by reference rather than by value. Only valid with Intent In. Intent Out and Intent InOut are automatically passed by reference. If checked, the argument represents the return value of the method. Only valid with Intent Out. If checked, then the argument is an array rather than scalar argument. When checked, fill in the Array Description fields to describe the shape of the array. If checked, then the argument is optional. Optional arguments are passed using the Variant data type. These fields describe the shape of an array argument. They are enabled when the Array argument field is checked.
Instance Type Properties: Module Constructor The name used for the module defined in the UclassnameTY.f90 file. The name used for the class constructor defined in the UclassnameTY.f90 file.
Page 20 of 41
Destructor
The name used for the class destructor defined in the UclassnameTY.f90 file.
Modifying the Generated Code After you either click the Save button to dismiss the Fortran COM Server Wizard or open the project workspace, the project is displayed, including the TODO.TXT file. The TODO.TXT file informs you of changes made to your project by the Wizard, and lets you know when you need to make changes. After the initial invocation of the Wizard, TODO.TXT lists the files that were added to the project. The source files are added to two folders in your FileView pane:
l
Source files that you will need to modify are placed in the Source Files folder. Files that are totally generated by the Wizard are placed in the Do Not Edit folder. It is futile to modify the files in the Do Not Edit folder, since these files are regenerated each time you invoke the Wizard and change the definition of the server (edits you make to these files will be lost).
Click: 1. The plus sign (+) at the Adder files line 2. The plus sign (+) at the Source Files folder The project screen appears as follows:
Page 21 of 41
The two files that you need to modify with a text editor are in the Source Files folder:
l
UAddingMachineTY.f90 contains a module that defines your class derivedtype. UIAdd.f90 contains the implementation of the IAdd methods.
We will first modify the file UAddingMachineTY.f90. To edit the file UAddingMachineTY.f90, either double-click its file name in the FileView pane or use the Open menu item in the File menu. The original file UAddingMachineTY.f90 contains the following code:
Page 22 of 41
A file named Uclass-nameTY.f90 is created for each class defined by the server. The file contains a module named AddingMachine_USE (in the form classname_USE). There are three places in this module where you may need to add code specific to your class:
l
The first entry in the module is your class derived-type. Initially it contains an "integer dummy" field to allow the module to be compiled without error. If your class has per-object data, remove the "integer dummy" field line and add your data to the derived-type. For the AddingMachine class, we add the following where the object stores the current value:
real(4) CurrentValue
The module also contains two module procedures by the names AddingMachine_CONSTRUCTOR and AddingMachine_DESTRUCTOR (referred to as classname_CONSTRUCTOR and classname_DESTRUCTOR below).
l
The classname_CONSTRUCTOR procedure is called immediately after an instance of the class derived-type is created because of the creation of a new object. This function is where you initialize the fields of the class derived-type, if necessary. The new derived-type is passed as an argument to the function. For the AddingMachine class, we initialize the current value to 0 by adding the following statement:
ObjectData%CurrentValue = 0
The classname_DESTRUCTOR procedure is called immediately before an instance of the class derived-type is destroyed because an object is being destroyed. This function is where you release any resources used by the fields of the class derived-type, if necessary. The derived-type is passed as an argument to the function. For the AddingMachine class, there is nothing that needs to be added.
We will now modify the other source file UIAdd.f90. Either double-click its file name in the FileView pane or use the File menu, Open item. The original file UIAdd.f90 contains the following code:
Page 23 of 41
A file by the name Uinterfacename.f90 (for example UIadd.f90) is created for each interface defined by the class. The file contains the methods of the class. Each method is named interfacename_methodname, for example: "IAdd_Clear". Each method is a function that is passed the class derived-type as the first argument. This gives the function access to the per-object data. Each function returns a 32-bit COM status code called an HRESULT. S_OK is a parameter that defines a success status. For additional information on COM status codes, see COM Status Codes: HRESULT. You replace the "! TODO: Add implementation" line in each method with the code for the method. For the IAdd interface, below is the implementation of its three methods:
IAdd_Clear: IAdd_Add: ObjectData%CurrentValue = 0 ObjectData%CurrentValue = ObjectData%CurrentValue + Operand
Save the file and from the Build menu, click Build Adder.dll to build the server. The COM server is now complete. Build Notes The Developer Studio project created by the Fortran AppWizard performs two additional steps not done in a typical Fortran project:
l
The build uses the MIDL compiler to compile the IDL file that describes the
Page 24 of 41
server. The MIDL compiler creates the server's Type library and generates C files that can be used in MIDL-based Marshalling (see Marshalling, Proxies and Stubs). The MIDL compiler uses the Visual C++ preprocessor by default during compilation. If the AppWizard determined that Visual C++ was not installed, it sets the MIDL options to use the Fortran preprocessor FPP instead. The options added are:
/cpp_cmdfpp /cpp_opt
l
A custom build step is added to register the COM server on your system. For a DLL COM server, registration is done using the REGSVR32 program. REGSVR32 could fail if the Window system directory is not on the Developer Studio's Executables Files list. On Windows NT and Windows 2000 systems, this is the \Winnt\System32 directory. On Windows 9x systems, this is typically the \Windows\System directory. If the registration step fails, do the following: 1. 2. 3. 4. In the Tools menu, click Options. Scroll to the Directories tab. Select the Executable files from the list. Add the windows system directory at the bottom of the list of directories. 5. Click OK
For an EXE COM server, the COM server itself is run with the /REGSERVER command-line option. Visual C++ ClassView If you also have Microsoft Visual C++ Version 6 installed on your system, you will notice that your server definition also appears in the Visual C++ ClassView pane. This occurs because the Visual C++ ClassView software can read and understand the IDL file created by the Fortran COM Server wizard. Note: Do not attempt to use the Visual C++ ClassView pane to modify the your server's definition. Always use the Fortran COM Server wizard to modify classes created by the Fortran COM Server wizard. Using Example Clients The server is now ready to be called by any COM client. The completed Adder sample is provided in the ...Df98\Samples\Advanced\COM\Adder directory. This directory also contains three clients written in Visual Fortran, Visual Basic, and Visual C++.
Page 25 of 41
To build the sample clients, first build the Adder sample. The GUIDs used by the sample COM clients work with the Adder Sample (not the GUIDs created in the example). If you followed the example COM server AddingMachine in earlier sections (starting with Creating a Fortran COM Server Project), the AddingMachine server that you created is identical to the Adder sample except for the GUID's assigned to the AddingMachine object. The Fortran COM Server AppWizard always generates new GUIDs for a new project. Each client displays a dialog box that provides the user interface to the Adding Machine. Below are notes on each of the clients:
l
VFAdder, the Visual Fortran client The VFAdder client is created using the Fortran Windows Application AppWizard. VFAdder is a simple Dialog Based Application. The main dialog box is defined to have Clear and Add buttons, an edit box to enter the addend, and a static text field that displays the current value:
The Fortran Module Wizard creates the module (AddingMachine.f90) that defines the IAdd interface methods for use from Fortran. Code is added to the beginning of the WinMain routine to initialize COM and create an AddingMachine object:
call COMINITIALIZE(ret) call COMCREATEOBJECTBYGUID (CLSID_AddingMachine, CLSCTX_ALL, IID_IAdd, & & gAddingMachine, status) call Check_Status(status, " Unable to create AddingMachine object")
The subroutine added as the handler of the Add button (VFAdderAdd) calls the $IAdd_Add routine and the $IAdd_GetValue routine:
! Call the AddingMachine Add method lret = DlgGet(dlg, IDC_EDIT_ADDEND, text) read(text, *) addend status = $IAdd_Add(gAddingMachine, addend) call Check_Status(status, " Add method returned failure status") Set the Current Value text field status = $IAdd_GetValue(gAddingMachine, value)
Page 26 of 41
write(text, *) value call Check_Status(status, " GetValue method returned failure status") lret = DlgSet(dlg, IDC_CURRENTVALUE, text)
The subroutine added as the handler of the Clear button (VFAdderClear) calls the $IAdd_Clear routine in a similar way. Code is added at the end of the WinMain routine to uninitialize COM and release the AddingMachine object:
if (gAddingMachine /= NULL) then ret = COMReleaseObject(gAddingMachine) endif call COMUNINITIALIZE()
l
VBAdder, the Visual Basic V6 client VBAdder is created from a new Standard EXE project. The form is modified to be similar to the VFAdder dialog box above. The first step in using the AddingMachine object from Visual Basic is to select the References menu item from the Project menu. Find AddingMachine 1.0 Type Library in the list and select it. Visual Basic reads the information from the type library and becomes aware of the AddingMachine object, the IAdd interface, and its methods. These will appear in the choices that the Visual Basic editor makes available to you as you enter code for the project. The following code is added to the VBAdder project:
An instance of the AddingMachine object is created by adding the following line to the General Declarations area:
Dim Adder As New AddingMachine
Page 27 of 41
VCAdder, the Visual C++ V6 client VCAdder is created using the MFC AppWizard (exe). The Dialog based option is selected in the wizard. The dialog box controls are copied from the VFAdder dialog box and pasted into the VCAdder dialog box. The following statement is added to VCAdder.h in order to give the source files access to the AddingMachine object, the IAdd interface and its methods:
#import "..\Adder.dll"
A member variable is added to the CVCAdderApp application class to hold the pointer to the AddingMachine object:
AddingMachineLib::IAddPtr m_pIAdd;
Code is added to the beginning of the CVCAdderApp::InitInstance method to initialize COM and create an AddingMachine object:
CoInitialize(NULL); m_pIAdd.CreateInstance(__uuidof(AddingMachineLib::AddingMachine));
A member variable for the Addend edit field, m_Addend A member variable for the CurrentValue static text field, m_CurrentValue Member functions for the clicked message on the Add, Clear, and Exit buttons - OnAdd, OnClear, and OnExit
OnClear calls the Clear method and the GetValue method in a similar way. Code is added at the end of the CVCAdderApp::InitInstance method to
Creating a COM Server release the AddingMachine object and uninitialize COM:
m_pIAdd.Release(); CoUninitialize();
Page 28 of 41
For design considerations and notes on using Visual Basic and Visual C++ clients, see Visual Basic and Visual C++ Client Notes. Modifying Your COM Servers You can modify the server to change the properties of any entry in the hierarchy, add new entries to the hierarchy, move entries in the hierarchy, or remove entries from the hierarchy. You use the same user interface of the Fortran COM Server Wizard that you used to create the COM server, simplifying any additions, deletions, or changes to your COM server definition. However, you need to pay careful attention to the following files when you modify the COM server definition:
l
View the TODO.TXT file. The TODO.TXT serves as a log of the changes you have made to the server definition. You may delete entries from TODO.TXT that are no longer useful to you. TODO.TXT is simply a text file that the wizard appends messages to. New versions of the source files in the Source files folder generated by the Fortran COM Server Wizard do not overwrite existing source files, since you may have modified existing source files. New source files are in the form of the original file name with a plus sign (+) appended, such as UIADD+.f90. Use the text editor to copy-and-paste the newly generated definitions into your previously modified source code (described below), as instructed by text in the TODO.TXT file.
The Fortran COM Server Wizard regenerates the files in the Do Not Edit folder whenever the server definition changes. To display the Fortran COM Server Wizard: 1. Start Developer Studio 2. Open a COM server project created by the Fortran COM Server AppWizard 3. Click the Fortran COM Server Wizard menu item in the View menu (see Using the Fortran COM Server Wizard to Define your COM Server) Once you have published an interface and clients are using it, COM rules state that you must not change the interface. Instead, you should create a new interface, in addition to the original one, and change the name. A common practice is to append a 2 to the name, for the second version of the interface. For example, IAdd would become IAdd2. The second version of your class should support both the original and new interfaces. Old clients continue to work
Page 29 of 41
using the original interface. New clients can use the new interface with the new functionality. The new interface has a different interface ID (IID) than the original interface. The text name is for the convenience of programmers. The IID uniquely identifies the interface. The Fortran COM Server Wizard adds messages to TODO.TXT anytime the Wizard makes a change to the project and anytime that you make a change to the server definition that requires you to manually edit one or more of the source files. For example, if you add a new method to the IAdd interface, SetValue, that sets the current value to a specific number, the Wizard generates new code to implement the new method in your server. Some of the new code is in the files in the Do Not Edit folder. The Wizard automatically changes these files. However, a new method skeleton needs to be added to the UIADD.f90 file in the Source Files folder. The wizard never creates a new file with the same name as a file in the Source Files folder, because that would cause the edits that you have made to the files to be lost. Instead, the wizard generates a file with a plus sign (+) appended to the end of the name, for example: UIADD+.f90. The file UIADD+.f90 contains the skeleton of the new method, IAdd_SetValue. In this case, the following message is added to TODO.TXT, and TODO.TXT is opened in the Developer Studio text editor:
The file UIAdd+.f90 has been generated with the following changes. You must merge these changes with the existing file UIAdd.f90. The method IAdd_SetValue has been added.
You must use a text editor to copy the new method skeleton from the UIADD+.f90 file and paste it into the UIADD.f90 file. Whenever a change is made to your server definition, "+" versions of the files in the Source Files folder are generated. You can ignore these files except when TODO.TXT informs you of an action you need to take. Other changes that would require you to manually merge changes into your source files include:
l l l l
Adding a new property. Changing the arguments of an existing method. Renaming a class, interface or method. Deleting a class, interface or method.
Page 30 of 41
This section provides information that should be considered when designing a Fortran COM server. It contains the following topics:
l l l
Method and Properties Data Types COM Status Codes: HRESULT Visual Basic and Visual C++ Client Notes
Method and Properties Data Types COM places some restrictions on the data types used in COM methods and properties. The reason for the restrictions is that COM can pass arguments between threads, processes and machines. This raises issues that are not present in older technologies, such as DLLs, that always run in the same address space as the caller. COM defines set of data types called Automation-compatible data types. These are the only data types that can be used in Automation and Dual interfaces. There are two advantages to restricting your COM interface to these data types:
l
Your server will be usable from clients written in the largest set of languages, including Visual Fortran, Visual Basic, and Visual C++. COM will automatically handle the passing of arguments between threads, processes and machines. This is called Type Library Marshalling. For more information, see Marshalling, Proxies and Stubs in Advanced COM Server Topics.
To restrict your server to Automation-compatible data types: 1. Select "Use only Automation data types" on the Interface property page. When defining a dual interface, this is automatically set. 2. Use only the following combinations of "Fortran data type" and "Interface data type" on the Argument property page. Note that Visual Fortran does not support the Currency, Decimal, or User Defined Type, Automation-compatible data types.
Creating a COM Server INTEGER(4) long SCODE Int IUnknown* IDispatch* float double DATE VARIANT_BOOL long unsigned char BSTR unsigned char
Page 31 of 41
INTEGER(INT_PTR_KIND ()) REAL(4) REAL(8) LOGICAL(2) LOGICAL(4) CHARACTER(1) CHARACTER(*) BYTE TYPE(VARIANT)
If you decide not to restrict your interface to Automation-compatible data types, the next approach is to restrict your interface to data types that can be described in the Interface Description Language (IDL). The Fortran COM Server Wizard automatically generates the IDL file from the description of your server. The MIDL compiler compiles the IDL file into a type library. MIDL can also automatically generate the code needed handle the passing of arguments between threads, processes and machines. Note, however, that a C compiler is required to use this option. For more information, see Marshalling, Proxies and Stubs in Advanced COM Server Topics. If you decide not to restrict your interface to IDL data types, your only remaining options are:
l
Implement Custom Marshalling - For more information, see Marshalling, Proxies and Stubs in Advanced COM Server Topics. Decide that you will never use your server across apartment, thread, or process boundaries. The server will not be usable in this manner because there is no way to pass the arguments across these boundaries.
Page 32 of 41
Each function returns a 32-bit COM status code called an HRESULT. An HRESULT is divided into fields:
l
The top bit (31) indicates whether the function succeeded or failed. The bit is set if the function failed. In Fortran, you can compare the HRESULT to < 0 as a test to see if a function failed. Bits 16 to 27 contain a facility code to indicate the facility that issued the HRESULT. Microsoft pre-defines a number of facility codes for its own use. If you create your own status codes, use FACILITY_ITF. All other facility codes are reserved by Microsoft. The first 16 bits, or low word, contain the error code specific to the error that occurred.
A typical HRESULT error value could be a value such as 0x80070057. The first hex digit, 8, indicates that bit 31 is set and that this is an error value. Bits 16 to 27 contain the value 7. This indicates the facility FACILITY_WIN32. The low word contains the value 0057. This is the specific code that identifies the error as E_INVALIDARG. To view the text description that corresponds to a system HRESULT value, use the Error Lookup tool in the Compaq Visual Fortran program folder. For example, entering the value 0x80070057 retrieves the text message "The parameter is incorrect ", as shown below:
You can also search for HRESULT values in the WINERROR.H file in the \VC\INCLUDE directory on the Visual Fortran CD-ROM. A COM server can also provide extended error information through the ErrorInfo objects. See Adding Support for COM ErrorInfo Objects. Visual Basic and Visual C++ Client Notes To use an object from Visual Basic, you must add a "reference" to the object to the Visual Basic project. Use the References item in the Project menu to display a list of the registered objects. Select the object in the list to inform Visual Basic that you will be using the object.
Page 33 of 41
Here are some points to be aware of when writing a server that can be used with Visual Basic clients:
l
l l l
Use only the Automation-compatible data types (see Method and Property Data Types). Arguments to a method can be passed ByVal, ByRef, or they can be the function return value. An argument to be passed ByVal, must be defined with Intent set to In. An argument to be passed ByRef, must be defined with Intent set to InOut. A function return value must be defined with Intent set to Out. It must have the Return Value field checked. The argument must be the final argument of the method. An array is always passed as a SafeArray ByRef. Therefore it must be defined with Intent set to InOut. To use an argument of the Visual Basic Boolean data type, set the Fortran data type to LOGICAL(2) and set the Interface data type to VARIANT_BOOL.
To use an object from Visual C++, use the #import directive. The syntax of the #import directive is:
#import "filename" [attributes] #import <filename> [attributes]
The filename is the name of the file containing the type library information. The directive makes the information in the type library available to your source file as a set of C++ classes. See the Visual C++ documentation for additional information and examples.
Choosing Between DLL or EXE COM Servers DLL Server Surrogates Discussion of Wizard Code Generation Adding Support for COM ErrorInfo Objects Threading Models Marshalling, Proxies and Stubs A Map of the Generated "Do Not Edit" Code
Choosing Between DLL or EXE COM Servers The basic tradeoff in choosing between a DLL (in-process) COM server and an EXE (out-of-process) COM server is one of performance vs. robustness:
Page 34 of 41
A DLL server provides the advantage of performance over an EXE server. Since the DLL server is loaded into the client's address space, there is less overhead involved in method calls. If the client code and the server object live in the same COM apartment, method calls are as efficient as DLL routine calls. An EXE server provides the advantage of robustness over a DLL server. Since the server object lives in a separate address space, the object cannot be affected by client memory handling bugs, and vice-versa. If the server crashes, the client doesn't necessarily crash as long as the client checks the results of all method calls and takes steps to recover from a "dead" object.
In addition to the tradeoff between performance and robustness, the following factors should also be considered:
l
With an EXE server, the object can run in a separate security context from the client. With a DLL server, the code of the object's methods executes using the client's access token. An EXE server can be run on a remote machine using COM's distributed object support.
You can load a DLL server into a "surrogate" to gain the benefits of an EXE server. This is explained in the next section. DLL Server Surrogates An in-process DLL server can be run in a separate process with the help of a surrogate. A surrogate runs as a separate process, loads the DLL server, and provides all of the mechanism that allows the DLL server to act as a local server. Windows provides a standard surrogate named DLLHOST.EXE. The primary advantage of using a surrogate is fault isolation. That is, if the server crashes it does not crash the client, and vice versa. The disadvantage is performance. There is significant performance overhead in executing methods in a separate application, as opposed to in a DLL. A DLL server is associated with a surrogate via entries in the system registry. This is done by associating the DLL server with an AppID. An AppID is a GUID. When using the standard surrogate, you can use the CLSID of a class in the DLL as the AppID, or a newly generated GUID. To generate a new GUID, use GUIDGEN.EXE in the Developer Studio Tools subdirectory: 1. Under the HKEY_CLASSES_ROOT\CLSID\{xxxxxxxx-xxxx-xxxx-xxxxxxxxxxxxxxx} key of the class, add an AppID entry with the value of the CLSID. Using the AddingMachine class as an example, the registry key would be HKEY_CLASSES_ROOT\CLSID\{904245FC-DD6D-11D3-98350000F875E193} and the AppID value would be {904245FC-DD6D-11D3-
Page 35 of 41
9835-0000F875E193}. 2. Under the HKEY_CLASSES_ROOT\AppID key, add a key using the AppID. Using the AddingMachine class as an example, the registry key would be HKEY_CLASSES_ROOT\AppID\{904245FC-DD6D-11D3-98350000F875E193}. Use the class name for the default value of the key, for example, "AddingMachine Class". Add a "DllSurrogate" entry with an empty string for the value. To use a surrogate, the system must be running Windows 2000, Windows NT 4.0 Service Pack 2 or later, Windows 98, Windows Me, or Windows 95 with the DCOM update. The client must request CLSCTX_LOCAL_SERVER rather than CLSCTX_INPROC_SERVER to use the surrogate rather than loading the DLL server in-process. Using a surrogate requires that the DLL server have a proxy/stub registered since the method invocations are between different processes. For information on proxies/stubs, see Marshalling, Proxies and Stubs. It is also possible to write a custom surrogate. See the online Win32 Platform SDK documentation for information. Discussion of Wizard Code Generation The Wizard generates the code for your project from the files in the subdirectory of your project named project-nametemplates. The project-name.hie file contains the definition of your COM server in an undocumented text language. You should not manually edit the project-name.hie file the Wizard does this for you. Most of the other files in the project-nametemplates directory are templates of the source files generated for your project. These templates contain source code that is copied "as-is" to the generated sources, and embedded directives that guide the Wizard in generating the code specific to the COM server that you define. The directives use the information in the project-name.hie file. The directives are undocumented and subject to change. When you create a new Fortran COM Server project, the AppWizard creates the project-nametemplates directory and copies the templates from the Visual Fortran COM Server Wizard templates directory, ...\Df98\Templates\COMServer. The files in the COM Server Wizard templates directory may change with each release of Visual Fortran, but the templates in your project-nametemplates directory are never automatically updated. For example, if you create a COM server using Visual Fortran Version 6.5 and the next release of Visual Fortran (such as Version 6.5A) contains updated templates, the templates for your COM server are not automatically updated to the new 6.5A templates. If you modify the definition of your server, your project continues to use the 6.5 templates
Page 36 of 41
that it was created with. This has the advantage of not introducing different code into a project that you have developed and tested. However, there are two cases where you may want to modify the templates that are used by your project:
l
A new release of Visual Fortran may contain additional features that can be used in your COM server project. Some of the new features may depend upon the new templates. These features will not be available to a preexisting project unless you update the templates in the projectnametemplates directory. You do this by copying all of the files in the Visual Fortran COM Server Wizard templates directory ...\Df98 \Templates\COMServer into the project-nametemplates directory, replacing all files with the same name. Your project will then use the new templates the next time that the definition of your server is modified and the project sources are regenerated. When you want to modify the code generated by the Wizard. You may edit a template in your project-nametemplates directory. Editing code that is copied "as-is" is straight forward. Attempting to modify the embedded directives is unsupported and could cause the Wizard to fail when using the modified template. The embedded directives begin with an @ (at-sign) character. The next character determines the type of directive. The end of many directives is also delimited by an @ character followed by the matching end character for the type of directive. For example, @[ and @] are the delimiters for one type of directive.
The advantage of modifying a template is that you can customize the code generated by the Wizard. The disadvantages of modifying a template are:
l
You must be very careful not to modify a template in a manner that causes the Wizard to generate bad code or fail. You will not be able to update the project templates with the templates from a new release of Visual Fortran without having to re-apply your modifications to the new templates.
Adding Support for COM ErrorInfo Objects COM supports ErrorInfo objects to allow a server to return more complete error information than is returned in an HRESULT. An ErrorInfo object returns the contextual description of the error, the source of the error, and the interface ID of the interface that originated the error. The ErrorInfo object can also include a pointer to an entry in a help file that provides help information on dealing with the error. To support ErrorInfo objects, your server must implement the ISupportErrorInfo interface. You can add a default implementation of this interface to your server:
Page 37 of 41
Select the class that you want to support ErrorInfo objects in the hierarchy pane. Click the right mouse button on the class to invoke the context-sensitive menu of the class and select the "Add Pre-defined Interface" menu item. Select ISupportErrorInfo in the "Select an Interface" dialog box and click OK.
ISupportErrorInfo contains a single method, InterfaceSupportsErrorInfo. The default implementation returns S_OK, which is sufficient if all of the interfaces of the class support ErrorInfo objects. Otherwise, you should modify the method to test the passed in interface ID and return S_OK for those interfaces that do support ErrorInfo objects, and E_FAIL for those that do not. When an interface that supports ErrorInfo objects is going to return an error status in HRESULT, it should also initialize an ErrorInfo object using the ICreateErrorInfo interface. See the ...\Df98\SAMPLES\ADVANCED\COM\ERRORINFO sample for an example of using ErrorInfo objects in Fortran. See the Platform SDK documentation for more information on ErrorInfo objects. Threading Models The Wizard supports two COM threading models for the classes that you create in a DLL COM server, Apartment and Single. The Wizard uses the Apartment threading model (also known as the Single Threaded Apartment model "STA") by default. The basic rules of the Apartment threading model are:
l
If two objects, A and B, are created in the same STA thread, and A is processing a method call, no other client can call either A or B until A completes. If B is created in a different thread from A, B can accept a method call while A is still processing and vice versa.
This means that if the class shares any global data among its objects, the global data must be protected from simultaneous access using thread synchronization primitives. This is because two instances of the class could be running in different threads. However the per-object instance data, that is, the fields in the class derived-type, are protected from simultaneous access by COM mechanisms. This is true except in the case where an object calls out to another object that triggers a reentrant invocation of the first object. A class using the Single threading model need not worry about simultaneous access to class global data, as well as per-object data. All objects of the class are created in a single thread and therefore only a single object of the class can be executing at any time. An EXE COM server generated by the wizard is single threaded. All method
Page 38 of 41
invocations are serialized by the server's message queue. Therefore, with an EXE server, you need not worry about simultaneous access to class global data, as well as per-object data. Explaining the COM threading models in detail is beyond the scope of this documentation. See the Win32 Platform SDK documentation for additional information. Marshalling, Proxies and Stubs COM supports the use of objects in separate processes (as when using an EXE server or a DLL surrogate), and the threading models described above, by the use of marshalling, proxies and stubs. This section presents an overview of marshalling, proxies and stubs. See the Win32 documentation for further information. Marshalling is the process of reading the parameters for a method call and preparing them for transmission to another execution context (for example, thread, process, or machine). Marshalling is done by a proxy. From the client's perspective, a proxy has the same interface as the object itself. The proxy's job is to make the object look like an object in the same execution context as the client. Proxies allow client code to be unconcerned about where the object actually lives. A proxy marshalls method parameters and transmits them to a stub associated with the object in the server. The stub unmarshalls the parameters and invokes the method in the server. From the server's perspective, this is no different than when it is called from a client in the same execution context. A server that is not an in-process DLL server always requires a proxy/stub pair. An in-process DLL server requires a proxy/stub pair when the client and object are in different apartments. There are three ways to assign a proxy/stub pair to a server: Type Library Marshalling: If you use only Automation-compatible data types in your methods and properties, COM can automatically use the "Universal Marshaller" as the proxy/stub for the server. The Universal Marshaller uses the description of the server in the type library to decide how to marshall the parameters. This is known as type library marshalling. Using type library marshalling requires no effort on your part, other than restricting your server to Automation-compatible data types.
Page 39 of 41
Your project uses the MIDL compiler to compile the IDL description of the server into a type library. At the same time, the MIDL compiler also generates the C source code necessary to build a proxy/stub DLL for the server. You must have a C compiler to build a proxy/stub DLL from the MIDL generated code. The proxy/stub DLL is itself an in-process DLL server and needs to be registered with the system. You would need to use MIDL-based marshalling if your server uses nonAutomation-compatible data types, and is used by a client in a different execution context. Your server can implement its own marshalling by implementing the IMarshall interface. This approach typically involves a lot of work and is done for performance reasons.
Custom Marshalling:
A Map of the Generated "Do Not Edit" Code This section presents an overview of which parts of the COM server functionality are implemented in which of the "Do Not Edit" source files generated by the wizard: File Name server.idl Description Contains the IDL description of the server. It is compiled by the MIDL compiler to produce the server's type library. Contains the global data and functions for the server. Contains the exported functions that are required of all COM Server DLLs. This includes DllMain, DLLRegisterServer, DLLUnregisterServer, DllGetClassObject, and DllCanUnloadNow. Declares the exported functions for the linker. Contains the main entry point of an EXE server. It also processes the command-line argument. Contains helper functions for the server. Contains definitions of the IClassFactory interface that is used to create instances of the classes defined by the server.
Page 40 of 41 Contains methods of the IClassFactory interface that is used to create instances of the classes defined by the server. Defines a module that contains definitions of parameters and types used in the implementation of the class. It also contains the implementation of the IUnknown methods of the class. A separate instance of this file is generated for each class defined in the server. Defines a module that contains the Fortran interfaces of the methods in the interface. It also contains the implementation of the Fortran "wrappers" that are called directly from the class' VTBL and call the methods implemented by the user. A separate instance of this file is generated for each interface defined in the class.
classnameTY.f90
interfacename.f90
To register a DLL server, use the REGSVR32.EXE system tool, a command-line tool. To register the DLL server, specify the path to the DLL and its file name (dll_path) by typing the following command:
REGSVR32 dll_path
The file REGSVR32.EXE is in your system directory, such as the \Winnt\System32 on Windows NT or Windows 2000 systems or \Windows\System on Windows 9x systems. If this directory is not in your PATH, include its path before the REGSVR32 command.
To register a an EXE server, run the server with the /REGSERVER command-line opton. For example:
exe_path /REGSERVER
3. Use the VFRUN tool to install the necessary run-time DLLs and related files.
Page 41 of 41
The VFRUN tool provides the run-time redistributable files for Visual Fortran applications, in the form of a self-installing kit. VFRUN is described in the Compaq Visual Fortran Installing and Getting Started and in the release notes. You can download the run-time redistributable files VFRUN kit from the Visual Fortran Web page (click Downloads area): http://www.compaq.com/fortran If you are using MIDL-based Marshalling or Custom Marshalling, then the marshalling DLL also needs to be installed and registered. For information on marshalling, see Marshalling, Proxies and Stubs in Advanced COM Server Topics.
Page 1 of 55
Call existing code that is written in another language Use procedures that may be difficult to implement in a particular language Gain advantages in processing speeds
Mixed-language programming is possible among the 32-bit languages Visual Fortran, Visual C/C++, Visual Basic, and MASM. Mixed-language programming in Win32 is different from that in 16-bit environments, and in many respects it is easier. To properly create mixed-language programs, rules must be established for naming variables and procedures, for stack use, and for argument passing among routines written in different languages. These rules, as a whole, are the calling convention. A calling convention includes:
l
Stack considerations Does a routine receive a varying or fixed number of arguments? Which routine clears the stack after a call? Naming conventions Is lowercase or uppercase significant or not significant? Are external names altered? Argument passing protocol Are arguments passed by value or by reference? What are the equivalent data types and data structures among languages?
This section provides information on the calling conventions available when writing routines written in Fortran, C, Visual C++, Visual Basic, and ia32 assembly language. It is organized into the following topics:
l l l l l l
Overview of Mixed-Language Issues Exchanging and Accessing Data in Mixed-Language Programming Handling Data Types in Mixed-Language Programming Visual Fortran/Visual C++ Mixed-Language Programs Fortran/Visual Basic Mixed-Language Programs Fortran/MASM Mixed-Language Programs
Page 2 of 55
Mixed-language programming involves a call from a routine written in one language to a function, procedure, or subroutine written in another language. For example, a Fortran main program may need to execute a specific task that you want to program separately in an assembly-language procedure, or you may need to call an existing DLL or system procedure. Mixed-language programming is possible with Visual Fortran, Visual C/C++, Visual Basic, and assembly language (MASM) because each language implements functions, subroutines, and procedures in approximately the same way. The following table shows how different kinds of routines from each language correspond to each other. For example, a C main program could call an external void function, which is actually implemented as a Fortran subroutine. Language Equivalents for Calls to Routines Language Fortran C and Visual C++ Visual Basic Assembly language Call with Return Value FUNCTION function Function Procedure Call with No Return Value SUBROUTINE (void) function Sub Procedure
There are some important differences in the way languages implement routines. Argument passing, naming conventions and other interface issues must be thoughtfully and consistently reconciled between any two languages to prevent program failure and indeterminate results. However, the advantages of mixedlanguage programming often make the extra effort worthwhile. A summary of a few mixed-language advantages and restrictions follows:
l
Fortran/Assembly Language Assembly-language routines are small and execute very quickly because they do not require initialization as do high-level languages like Fortran and C. Also, they allow access to hardware instructions unavailable to the highlevel language user. In a Fortran/assembly-language program, compiling the main routine in Fortran gives the assembly code access to Fortran highlevel procedures and library functions, yet allows freedom to tune the assembly-language routines for maximum speed and efficiency. The main program can also be an assembly-language program.
Page 3 of 55
Fortran/Visual Basic A mix of Fortran and Visual Basic 4.0 or higher (32-bit) allows you to use the easy-to-implement user-interface features of Visual Basic, yet do all your computation, especially floating-point math, in Fortran routines. In a Fortran/Visual Basic program, the main routine must be Visual Basic. It is not possible to call Basic routines from Fortran. You can call Visual Basic routines as callbacks, but the main program must be Visual Basic.
Fortran/C (or C++) Generally, Fortran/C programs are mixed to allow one to use existing code written in the other language. Either Fortran or C can call the other, so the main routine can be in either language. To use the same Microsoft visual development environment for multiple languages, you must have the same version of the visual development environment for your languages (see Mixed-Language Development Support in Compaq Visual Fortran Installing and Getting Started).
This section provides an explanation of the keywords, attributes, and techniques you can use to reconcile differences between Fortran and other languages. Adjusting calling conventions, adjusting naming conventions and writing interface procedures are discussed in the next sections:
l l l
Adjusting Calling Conventions in Mixed-Language Programming Adjusting Naming Conventions in Mixed-Language Programming Prototyping a Procedure in Fortran
After establishing a consistent interface between mixed-language procedures, you then need to reconcile any differences in the treatment of individual data types (strings, arrays, and so on). This is discussed in Exchanging and Accessing Data in Mixed-Language Programming . Note: This section uses the term "routine" in a generic way, to refer to functions, subroutines, and procedures from different languages.
Page 4 of 55
In a mixed-language program, different languages cannot share the same header files. If, as a result, you link Fortran and C routines that use different calling conventions, the error is not apparent until the bad call is made at runtime. During execution, the bad call causes indeterminate results and/or a fatal error, often somewhere in the program that has no apparent relation to the actual cause: memory/stack corruption due to calling errors. Therefore, you should check carefully the calling conventions for each mixed-language call. The discussion of calling conventions between languages applies only to external procedures. You cannot call internal procedures from outside the program unit that contains them. A calling convention affects programming in five ways: 1. The caller routine uses a calling convention to determine the order in which to pass arguments to another routine; the called routine uses a calling convention to determine the order in which to receive the arguments passed to it. In Fortran, you can specify these conventions in a mixedlanguage interface with the INTERFACE statement or in a data or function declaration. 32-bit Visual C/C++ and Fortran both pass arguments in order from left to right. 2. The caller routine and the called routine use a calling convention to determine which of them is responsible for adjusting the stack in order to remove arguments when the execution of the called routine is complete. You can specify these conventions with ATTRIBUTES (cDEC$ ATTRIBUTES compiler directive) options such as C or STDCALL. 3. The caller routine and the called routine use a calling convention to select the option of passing a variable number of arguments. 4. The caller routine and the called routine use a calling convention to pass arguments by value (values passed) or by reference (addresses passed). Individual Fortran arguments can also be designated with ATTRIBUTES option VALUE or REFERENCE. 5. The caller routine and the called routine use a calling convention to establish naming conventions for procedure names. You can establish any procedure name you want, regardless of its Fortran name, with the ALIAS directive (or ATTRIBUTES option ALIAS). This is useful because C anmd Basic are case sensitive, while Fortran is not. Specific calling-convention issues are discussed in the following sections:
l l l l l
ATTRIBUTES Properties and Calling Conventions Stack Considerations in Calling Conventions Fortran/C Calling Conventions Fortran/Visual Basic Calling Conventions Fortran/MASM Calling Conventions
Page 5 of 55
The C, STDCALL, REFERENCE, and VARYING properties for an entire routine. The VALUE and REFERENCE properties for individual arguments.
By default, Fortran passes all data by reference (except the hidden length argument of strings, which is passed by value). If the C or STDCALL option is used, the default changes to passing almost all data except arrays by value. However, in addition to the calling-convention options C and STDCALL, you can specify argument options, VALUE and REFERENCE, to pass arguments by value or by reference, regardless of the calling convention option. Arrays can only be passed by reference. Different Fortran calling conventions can be specified by declaring the Fortran procedure to have certain attributes. For example, on ia32 systems:
INTERFACE SUBROUTINE MY_SUB (I) !DEC$ ATTRIBUTES C, ALIAS:'_My_Sub' :: MY_SUB INTEGER I END SUBROUTINE MY_SUB END INTERFACE
! ia32 systems
This code (on ia32 systems) declares a subroutine named MY_SUB with the C property and the external name _My_Sub set with the ALIAS property. On ia64 systems, there is no leading underscore for external names like MY_SUB, so the correct !DEC$ ATTRIBUTES line is:
!DEC$ ATTRIBUTES C, ALIAS:'My_Sub' :: MY_SUB ! ia64 systems
To write code for both ia32 and ia64 platforms, use the conditional compilation features of the IF Directive Construct, perhaps using the predefined preprocessor macros listed under the /define option (such as _M_IX86 and _M_IA64). The following table summarizes the effect of the most common Fortran callingconvention directives. Calling Conventions for ATTRIBUTES Options
Page 6 of 55
Default Argument Scalar Scalar [value] Scalar [reference] String Reference Value Reference Reference, either Len:Mixed or Len:End Error Reference, either Len:Mixed or No Len Reference Error Reference Reference
STDCALL
C, REFERENCE
STDCALL, REFERENCE
Reference Value Reference Reference, either Len:Mixed or Len:End String(1:1) Reference, No Len
Reference Value Reference Reference, either Len:Mixed or Len:End String(1:1) Reference, No Len
Array Array [value] Array [reference] Derived Type Derived Type [value] Derived Type [reference]
Reference Error Reference Value, size dependent Value, size dependent Reference
Reference Error Reference Value, size dependent Value, size dependent Reference
Programming with Mixed Languages F90 Pointer F90 Pointer [value] F90 Pointer [reference] Descriptor Error Descriptor Error Descriptor Error Descriptor Error
Descriptor
Descriptor
Descriptor
Descriptor
Descriptor
Procedure Name Suffix Case Stack Cleanup @n (ia32 systems) Upper Case Callee none Lower Case Caller @n (ia32 systems) Lower Case Callee none Lower Case Caller @n (ia32 systems) Lower Case Callee
The terms in the above table mean the following: [value] [reference] Value Reference Argument assigned the VALUE attribute. Argument assigned the REFERENCE attribute. The argument value is pushed on the stack. All values are padded to the next 4-byte boundary. On ia32 systems, the 4-byte argument address is pushed on the stack. On ia64 systems, the 8-byte argument address is pushed on the stack. For certain string arguments:
l
Len:Mixed or Len:End
Len:Mixed applies when /iface:mixed_str_len_arg is set. The length of the string is pushed (by value) on the stack immediately after the address of the beginning of the string. Len:End applies when /iface:nomixed_str_len_arg is set. The length of the string is pushed (by value) on the stack after all of the other arguments.
Programming with Mixed Languages Len:Mixed or No Len For certain string arguments:
l
Page 8 of 55
Len:Mixed applies when /iface:mixed_str_len_arg is set. The length of the string is pushed (by value) on the stack immediately after the address of the beginning of the string. No Len applies when /iface:nomixed_str_len_arg is set. The length of the string is not available to the called procedure.
No Len String(1:1)
For string arguments, the length of the string is not available to the called procedure. For string arguments, the first character is converted to INTEGER(4) as in ICHAR(string(1:1)) and pushed on the stack by value. Produces a compiler error. On ia32 systems, the 4-byte address of the array descriptor. On ia64 systems, the 8-byte address of the array descriptor. On ia32 systems, the at sign (@) followed by the number of bytes (in decimal) required for the argument list. On ia32 systems, derived-type arguments specified by value are passed as follows:
l l
Arguments from 1 to 4 bytes are passed by value. Arguments from 5 to 8 bytes are passed by value in two registers (two arguments). Arguments more than 8 bytes provide value semantics by passing a temporary storage address by reference.
Procedure name in all uppercase. Procedure name in all lowercase. The procedure being called is responsible for removing arguments from the stack before returning to the caller. The procedure doing the call is responsible for removing arguments from the stack after the call is over.
The following table shows which Fortran ATTRIBUTES options match other
Programming with Mixed Languages language calling conventions. Matching Calling Conventions Other Language Calling Convention Visual C/C++ cdecl (default) Visual C/C++ __stdcall Visual Basic Visual Basic CDECL keyword MASM C (in PROTO and PROC declarations) MASM STDCALL (in PROTO and PROC declarations)
Page 9 of 55
The ALIAS option can be used with any other Fortran calling-convention option to preserve mixed-case names. You can also use the DECORATE option with ALIAS option so the external name specified in ALIAS has prefix and postfix decorations performed on it that are associated with the calling mechanism that is in effect. Note: When interfacing to the Windows graphical user interface or making API calls, you will typically use STDCALL. See Creating Windows Applications for more information on Windows programming.
Page 10 of 55
You can call routines with a variable number of arguments by including the ATTRIBUTES C and VARYING options in your interface to a routine. The VARYING option prevents Fortran from enforcing a matching number of arguments in routines. The VARYING option is not necessary with intrinsic Fortran 90 routines with optional arguments, where argument order and/or keywords determine which arguments are present and which are absent. In MASM, stack control is also set by the C or STDCALL convention declared for the procedure, but you can write MASM code to control the stack within the procedure any way you wish. In addition, you can specify the USES option in the PROC directive to save and restore certain registers automatically.
Alternatively, instead of changing the calling convention of the C code, you can adjust the Fortran source code by using the C option. This is set with the ATTRIBUTES directive. For example, the following declaration assumes the subroutine is called with the C calling convention:
SUBROUTINE CALLED_FROM_C (A) !DEC$ ATTRIBUTES C :: CALLED_FROM_C INTEGER A
If a varying number of arguments are to be passed, the C and VARYING properties are needed in the Fortran procedure definition and the CDECL keyword needed in the Basic DECLARE statement in order to establish the
Page 11 of 55
C calling and naming convention. When passing character arguments, the Fortran routine must not pass the hidden length of character arguments, such as by using the ATTRIBUTES property STDCALL for the routine and specifying the REFERENCE property for any character arguments. Since STDCALL also lowercases Fortran names, the Fortran subprogram name should be referenced in lowercase from the Visual Basic program.
The following Fortran and Visual Basic statements establish an example Fortran function to be called from Basic:
! Fortran Subprogram establishing Fortran function. INTERFACE DOUBLE PRECISION FUNCTION GetFVal (r1) !DEC$ ATTRIBUTES ALIAS:'GetFVal' :: GetFVal !DEC$ ATTRIBUTES VALUE :: r1 REAL r1 END FUNCTION END INTERFACE
'FORM.FRM Basic Form to establish Fortran function Declare Function GetFVal Lib "C:\f90\FVAL.DLL" (ByVal r1 As Single) As Double
Page 12 of 55
The following Fortran and MASM statements set up a Fortran-callable MASM function using the C calling convention:
! Fortran C interface prototype INTERFACE INTEGER FUNCTION Forfunc (I1, I2) !DEC$ ATTRIBUTES C, ALIAS:'Forfunc' :: Forfunc INTEGER I1 INTEGER(2) I2 END INTERFACE WRITE(*,*) Forfunc (I1, I2) END ! End Fortran C interface ;MASM C PROTOTYPE .MODEL FLAT, C Forfunc PROTO C, forint:SDWORD, shorti: ptr SWORD .CODE Forfunc PROC C, forint:SDWORD, shorti: ptr SWORD ... Forfunc ENDP END
Visual C/C++ and Visual Basic Naming Conventions MASM Naming Conventions Naming Conventions for Fortran, C, Visual C++, Visual Basic, and MASM Reconciling the Case of Names Fortran Module Names and ATTRIBUTES
Page 13 of 55
Visual C/C++ and Visual Basic preserve case sensitivity in their symbol tables while Fortran by default does not, a difference that requires attention. Fortunately, you can use the Fortran directive ATTRIBUTES ALIAS option to resolve discrepancies between names, to preserve mixed-case names, or to override the automatic conversion of names to all uppercase by the Fortran default naming, or the automatic conversion to all lowercase by Fortran's STDCALL and C naming convention. Visual C++ uses the same calling convention and argument-passing techniques as C, but naming conventions are different because of Visual C++ decoration of external symbols. When the C++ code resides in a .cpp file (created when you select C/C++ file from the visual development environment), C++ name decoration semantics are applied to external names, often resulting in linker errors. The extern "C" syntax makes it possible for a Visual C++ module to share data and routines with other languages by causing Visual C++ to drop name decoration. The following example declares prn as an external function using the C naming convention. This declaration appears in Visual C++ source code:
extern "C" { void prn(); }
To call functions written in Fortran, declare the function as you would in C and use a "C" linkage specification. For example, to call the Fortran function FACT from Visual C++, declare it as follows:
extern "C" { int __stdcall FACT( int n ); }
The extern "C" syntax can be used to adjust a call from Visual C++ to other languages, or to change the naming convention of Visual C++ routines called from other languages. However, extern "C" can only be used from within Visual C++. If the Visual C++ code does not use extern "C" and cannot be changed, you can call Visual C++ routines only by determining the name decoration and generating it from the other language. Such an approach should only be used as a last resort, because the decoration scheme is not guaranteed to remain the same between versions. Use of extern "C" has some restrictions:
l l
You cannot declare a member function with extern "C". You can specify extern "C" for only one instance of an overloaded function; all other instances of an overloaded function have Visual C++ linkage.
For more information on the extern "C" linkage specification, see the Microsoft Visual C++ Language Reference.
Page 14 of 55
CASEMAP: NONE (equivalent to /Cx) preserves the case of identifiers in PUBLIC, COMM, EXTERNDEF, EXTERN, PROTO, and PROC declarations. CASEMAP: NOTPUBLIC (equivalent to /Cp) preserves the case of all user identifiers; this is the default. CASEMAP: ALL (equivalent to /Cu) translates all identifiers to uppercase.
Naming Conventions for Fortran, C, Visual C++, Visual Basic, and MASM
The following table summarizes how Fortran, Visual C/C++, Visual Basic and MASM handle procedure names. Note that for MASM, the table does not apply if the CASEMAP: ALL option is used. Naming Conventions in Fortran, C, Visual C++, Visual Basic, and MASM Language Fortran Fortran Fortran C C Visual C++ Visual Basic Attributes cDEC$ ATTRIBUTES C cDEC$ ATTRIBUTES STDCALL default cdecl (default) __stdcall Default Default Name Translated As _name _name@n _name@n _name _name@n _name@@decoration _name@n Case of Name in .OBJ File All lowercase All lowercase All uppercase Mixed case preserved Mixed case preserved Mixed case preserved Mixed case preserved
Programming with Mixed Languages MASM MASM C (in PROTO and PROC declarations) STDCALL (in PROTO and PROC declarations) _name _name@n
The leading underscore (such as _name) is used on ia32 systems only (not on ia64 systems). @n represents the stack space, in decimal notation, occupied by parameters on ia32 only (not on ia64 systems).
Each integer occupies 4 bytes, so the symbol name placed in the .OBJ file on ia32 systems is:
_Sum_Up@12
On ia64 systems, the symbol name placed in the .OBJ file is:
Sum_Up
All-Uppercase Names If you call a Fortran routine that uses Fortran defaults and cannot recompile the Fortran code, then in C and Visual Basic you must use an alluppercase name to make the call. In MASM you must either use an alluppercase name or set the OPTION CASEMAP directive to ALL, which translates all identifiers to uppercase. Use of the __stdcall convention in C code or STDCALL in MASM PROTO and PROC declarations is not enough, because __stdcall and STDCALL always preserve case in these languages. Fortran generates all-uppercase names by default and the C or MASM code must match it. For example, these prototypes establish the Fortran function FFARCTAN (angle) where the argument angle has the ATTRIBUTES VALUE property:
In C:
Page 16 of 55
In Visual Basic:
In MASM:
.MODEL FLAT, STDCALL FFARCTAN PROTO STDCALL, angle: REAL4 ... FFARCTAN PROC STDCALL, angle: REAL4
All-Lowercase Names If the name of the routine appears as all lowercase in C or MASM, then naming conventions are automatically correct when the C or STDCALL option is used in the Fortran declaration. Any case may be used in the Fortran source code, including mixed case since the C and STDCALL options change the name to all lowercase. You cannot call a Visual Basic routine from Fortran directly (see Fortran/Visual Basic Mixed-Language Programs), so Basic routine names are never translated.
Mixed-Case Names If the name of a routine appears as mixed-case in C or MASM and you cannot change the name, then you can resolve this naming conflict by using the Fortran ATTRIBUTES ALIAS option. ALIAS is required in this situation because otherwise Fortran will not preserve the mixed-case name. To use the ALIAS option, place the name in quotation marks exactly as it is to appear in the .OBJ file. The following is an example on ia32 systems for referring to the C function My_Proc:
!DEC$ ATTRIBUTES ALIAS:'_My_Proc' :: My_Proc
On ia64 systems, this would be coded without the leading underscore as:
!DEC$ ATTRIBUTES ALIAS:'My_Proc' :: My_Proc
Page 17 of 55
_MODULENAME_mp_ENTITY [ @stacksize ] MODULENAME is the name of the module and is all uppercase by default. ENTITY is the name of the module procedure or module data contained within MODULENAME. ENTITY is also uppercase by default. _mp_ is the separator between the module and entity names and is always lowercase. For example:
MODULE mymod INTEGER a CONTAINS SUBROUTINE b (j) INTEGER j END SUBROUTINE END MODULE
This results in the following symbols being defined in the compiled .OBJ file on ia32 systems:
_MYMOD_mp_A _MYMOD_mp_B@4
Compiler options can affect the naming of module data and procedures. Note: Except for ALIAS, ATTRIBUTES options do not affect the module name, which remains uppercase. The following table shows how each ATTRIBUTES option affects the subroutine in the previous example module. Effect of ATTRIBUTES options on Fortran Module Names ATTRIBUTES Option Given to Routine 'b' None C STDCALL Procedure Name in .OBJ file on ia32 Systems _MYMOD_mp_B@4 _MYMOD_mp_b _MYMOD_mp_b@4 Procedure Name in .OBJ file on ia64 Systems MYMOD_mp_B MYMOD_mp_b MYMOD_mp_b
Programming with Mixed Languages ALIAS Overrides all others, name as given in the alias No effect on name
Page 18 of 55 Overrides all others, name as given in the alias No effect on name
VARYING
You can write code to call Fortran modules or access module data from other languages. As with other naming and calling conventions, the module name must match between the two languages. Generally, this means using the C or STDCALL convention in Fortran, and if defining a module in another language, using the ALIAS option to match the name within Fortran. Examples are given in the section Using Modules in Mixed-Language Programming.
Page 19 of 55
The Fortran call to this function should be declared with the following INTERFACE block on ia32 systems:
INTERFACE SUBROUTINE my_Proc (I) !DEC$ ATTRIBUTES C, ALIAS:'_My_Proc' :: my_Proc INTEGER I END SUBROUTINE my_Proc END INTERFACE
Note that:
l
On ia64 systems, the leading underscore in _My_Proc should be omitted. The ATTRIBUTES line on ia64 systems contains:
!DEC$ ATTRIBUTES C, ALIAS:'My_Proc' :: my_Proc
Except in the ALIAS string, the case of My_Proc in the Fortran program does not matter.
Passing Arguments in Mixed-Language Programming Using Modules in Mixed-Language Programming Using Common External Data in Mixed-Language Programming
Generally, if you have a large number of parameters to work with or you have a large variety of parameter types, you should consider using modules or external data declarations. This is true when using any given language, and to an even greater extent when using mixed languages.
By value, which passes the argument's value. By reference, which passes the address of the arguments. On ia32 systems, Fortran, Visual Basic, C, and Visual C++ use 4-byte addresses. On ia64 systems, Fortran uses 8-byte addresses.
Page 20 of 55
You need to make sure that for every call, the calling program and the called routine agree on how each argument is passed. Otherwise, the called routine receives bad data. The Fortran technique for passing arguments changes depending on the calling convention specified. By default, Fortran passes all data by reference (except the hidden length argument of strings, which is passed by value). If the ATTRIBUTES C or STDCALL option is used, the default changes to passing all data by value except arrays. If the procedure has the REFERENCE option as well as the C or STDCALL option, all arguments by default are passed by reference. In Fortran, in addition to establishing argument passing with the callingconvention options C and STDCALL, you can specify argument options, VALUE and REFERENCE, to pass arguments by value or by reference. In mixedlanguage programming, it is a good idea to specify the passing technique explicitly rather than relying on defaults. Note: In addition to ATTRIBUTES, the compiler option /iface also establishes some default argument passing conventions (such as for hidden length of strings). Examples of passing by reference and value for C, Visual Basic and MASM follow. All are interfaces to the example Fortran subroutine TESTPROC below. The definition of TESTPROC declares how each argument is passed. The REFERENCE option is not strictly necessary in this example, but using it makes the argument's passing convention conspicuous.
SUBROUTINE TESTPROC( VALPARM, REFPARM ) !DEC$ ATTRIBUTES VALUE :: VALPARM !DEC$ ATTRIBUTES REFERENCE :: REFPARM INTEGER VALPARM INTEGER REFPARM END SUBROUTINE
l
Fortran/C example of arguments passed by value and reference In C and Visual C++ all arguments are passed by value, except arrays, which are passed by reference to the address of the first member of the array. Unlike Fortran, C and Visual C++ do not have calling-convention directives to affect the way individual arguments are passed. To pass nonarray C data by reference, you must pass a pointer to it. To pass a C array by value, you must declare it as a member of a structure and pass the structure. The following C declaration sets up a call to the example Fortran TESTPROC subroutine:
Page 21 of 55
Fortran/Visual Basic example of arguments passed by value and reference In Visual Basic, arguments are passed by reference by default. To pass arguments by value, you use the keyword BYVAL in front of the argument in the DECLARE statement. For example:
Declare Sub TESTPROC Lib "C:\f90\TESTPROC.DLL" (ByVal Valparm As Long, Refparm As Long)
Strings are a special case (see the discussion on character strings in Handling Character Strings). Arrays of numbers require special handling (see data type considerations in Calling Visual Fortran from Visual Basic.
l
Fortran/MASM example of arguments passed by value and reference In MASM, arguments are passed by value by default. Arguments to be passed by reference are designated with PTR in the PROTO and PROC directives. For example:
TESTPROC PROTO STDCALL, valparm: SDWORD, refparm: PTR SDWORD
To use an argument passed by value, use the value of the variable. For example:
mov eax, valparm ; Load value of argument
This statement places the value of valparm into the EAX register. To use an argument passed by reference, use the address of the variable. For example:
mov ecx, refparm ; Load address of argument mov eax, [ecx] ; Load value of argument
These statements place the value of refparm into the EAX register. The following table summarizes how to pass arguments by reference and value. An array name in C is equated to its starting address because arrays are normally passed by reference. You can assign the REFERENCE property to a procedure, as well as to individual arguments. Passing Arguments by Reference and Value
Page 22 of 55
Language Fortran
ATTRIBUTE Default
Argument Type Scalars and derived types Scalars and derived types Arrays Arrays Non-arrays Arrays
Default
Cannot pass by value Cannot pass by value Default Struct {type} array_name ByVal Default
This table does not describe argument passing of strings and Fortran 95/90 pointer arguments in Visual Fortran, which are constructed differently than other arguments. By default, Fortran passes strings by reference along with the string length. String length placement depends on whether the compiler option /iface:mixed_str_len_arg (immediately after the address of the beginning of the string) or /iface:nomixed_str_len_arg (after all arguments) is set. Fortran 95/90 array pointers and assumed-shape arrays are passed by passing the address of the array descriptor. For a discussion of the effect of attributes on passing Fortran 95/90 pointers and strings, see Handling Fortran 90 Pointers and Allocatable Arrays and Handling Character Strings.
Page 23 of 55
The C code:
\* C code accessing module data *\ extern float EXAMP_mp_A[3]; extern int EXAMP_mp_I1, EXAMP_mp_I2; extern char EXAMP_mp_LINE[80]; extern struct { int N; char INFO[30]; } EXAMP_mp_MYDATA;
When the C++ code resides in a .cpp file (created when you select C/C++ file from the visual development environment), C++ semantics are applied to external names, often resulting in linker errors. In this case, use the extern "C" syntax (see Visual C/C++ and Visual Basic Naming Conventions):
\* C code accessing module data in .cpp file*\ extern "C" float EXAMP_mp_A[3]; extern "C" int EXAMP_mp_I1, EXAMP_mp_I2; extern "C" char EXAMP_mp_LINE[80]; extern "C" struct { int N; char INFO[30]; } EXAMP_mp_MYDATA;
You can also define a module procedure in C and make that routine part of a Fortran module by using the ALIAS directive. The C code:
// C procedure void pythagoras (float a, float b, float *c) { *c = (float) sqrt(a*a + b*b); }
Using the same example when the C++ code resides in a .cpp file, use the extern "C" syntax (see Visual C/C++ and Visual Basic Naming Conventions):
// C procedure extern "C" void pythagoras (float a, float b, float *c) { *c = (float) sqrt(a*a + b*b); }
Page 24 of 55
The Fortran code to call this routine using the module CPROC:
! Fortran 95/90 Module including procedure USE CPROC CALL PYTHAGORAS (3.0, 4.0, X) TYPE *,X END
Using Global Variables in Mixed-Language Programming A variable can be shared between Fortran and C or MASM by declaring it as global (or COMMON) in one language and accessing it as an external variable in the other language. Visual Basic cannot access another language's global data or share its own. In Fortran/Basic programs, variables must be passed as
Page 25 of 55
In Fortran, a variable can access a global parameter by using the EXTERN option for ATTRIBUTES. For example:
!DEC$ ATTRIBUTES C, EXTERN :: idata INTEGER idata (20)
EXTERN tells the compiler that the variable is actually defined and declared global in another source file. If Fortran declares a variable external with EXTERN, the language it shares the variable with must declare the variable global. In C, a variable is declared global with the statement:
int idata[20]; // declared as global (outside of any function)
MASM declares a parameter global (PUBLIC) with the syntax: PUBLIC [langtype] name where name is the name of the global variable to be referenced, and the optional langtype is STDCALL or C. The option langtype, if present, overrides the calling convention specified in the .MODEL directive. Conversely, Fortran can declare the variable global (COMMON) and other languages can reference it as external:
! Fortran declaring PI global REAL PI COMMON /PI/ PI ! Common Block and variable have the same name
Note that the global name C references is the name of the Fortran common block, not the name of a variable within a common block. Thus, you cannot use blank common to make data accessible between C and Fortran. In the preceding example, the common block and the variable have the same name, which helps keep track of the variable between the two languages. Obviously, if a common block contains more than one variable they cannot all have the common block name. (See Using Fortran Common Blocks and C Structures.) MASM can also access Fortran global (COMMON) parameters with the ATTRIBUTES EXTERN directive. The syntax is: EXTERN [langtype] name
Page 26 of 55
where name is the name of the global variable to be referenced, and the optional langtype is STDCALL or C. Using Fortran Common Blocks and C Structures To reference C structures from Fortran common blocks and vice versa, you must take into account the way the common blocks and structures differ in their methods of storing member variables in memory. Fortran places common block variables into memory in order as close together as possible, with the following rules:
l
A single BYTE, INTEGER(1), LOGICAL(1), or CHARACTER variable in common block list begins immediately following the previous variable or array in memory. All other types of single variables begin at the next even address immediately following the previous variable or array in memory. All arrays of variables begin on the next even address immediately following the previous variable or array in memory, except for CHARACTER arrays which always follow immediately after the previous variable or array. All common blocks begin on a four-byte aligned address.
Because of these padding rules, you must consider the alignment of C structure elements with Fortran common block elements and assure matching either by making all variables exactly equivalent types and kinds in both languages (using only 4-byte and 8-byte data types in both languages simplifies this) or by using the C pack pragmas in the C code around the C structure to make C data packing like Fortran's. For example:
#pragma pack(2) struct { int N; char INFO[30]; } examp; #pragma pack()
To restore the original packing, you must add #pragma pack( ) at the end of the structure. (Remember: Fortran module data can be shared directly with C structures with appropriate naming.) Once you have dealt with alignment and padding, you can give C access to an entire common block or set of common blocks. Alternatively, you can pass individual members of a Fortran common block in an argument list, just as you can any other data item. Use of common blocks for mixed-language data exchange is discussed in the following sections:
l
Page 27 of 55
Accessing Common Blocks and C Structures Directly You can access Fortran common blocks directly from C by defining an external C structure with the appropriate fields, and making sure that alignment and padding between Fortran and C are compatible. The C and ALIAS ATTRIBUTES options can be used with a common block to allow mixed-case names. As an example, suppose your Fortran code has a common block named Really, as shown:
!DEC$ ATTRIBUTES ALIAS:'Really' :: Really REAL(4) x, y, z(6) REAL(8) ydbl COMMON / Really / x, y, z(6), ydbl
You can access this data structure from your C code with the following external data structure:
#pragma pack(2) extern struct { float x, y, z[6]; double ydbl; } Really; #pragma pack()
You can also access C structures from Fortran by creating common blocks that correspond to those structures. This is the reverse case from that just described. However, the implementation is the same because after common blocks and structures have been defined and given a common address (name), and assuming the alignment in memory has been dealt with, both languages share the same memory locations for the variables. Passing the Address of a Common Block To pass the address of a common block, simply pass the address of the first variable in the block, that is, pass the first variable by reference. The receiving C or Visual C++ module should expect to receive a structure by reference. In the following example, the C function initcb receives the address of a common block with the first variable named n, which it considers to be a pointer to a structure with three fields: Fortran source code:
! INTERFACE SUBROUTINE initcb (BLOCK) !DEC$ ATTRIBUTES C :: initcb !DEC$ ATTRIBUTES REFERENCE :: BLOCK
Page 28 of 55
C source code:
// #pragma pack(2) struct block_type { int n; double x; double y; }; #pragma pack() // void initcb( struct block_type *block_hed ) { block_hed->n = 1; block_hed->x = 10.0; block_hed->y = 20.0; }
Programming with Mixed Languages CHARACTER(1) CHARACTER*(*) COMPLEX(4) unsigned char --BYTE
Page 29 of 55
See Handling Character Strings struct complex4 { float real, imag; }; struct complex8 { double real, imag; }; --COMPLEX4 STRUCT 4 real REAL4 0 imag REAL4 0 COMPLEX4 ENDS COMPLEX8 STRUCT 8 real REAL8 0 imag REAL8 0 COMPLEX8 ENDS
COMPLEX(8)
---
The following sections describe how to reconcile data types between the different languages:
l l l l l l
Numeric, Complex, and Logical Data Types Fortran 90 Array Pointers and Allocatable Arrays Compaq Fortran Pointers Arrays and Visual Fortran Array Descriptors Character Strings User-Defined Types
Page 30 of 55
Note: On ia32 systems, Fortran functions of type COMPLEX place a hidden COMPLEX argument at the beginning of the argument list. C functions that implement such a call from Fortran must declare this hidden argument explicitly, and use it to return a value. The C return type should be void. Following are the Visual C/C++ structure definitions for the Fortran COMPLEX types:
struct complex4 { float real, imag; }; struct complex8 { double real, imag; };
The MASM structure definitions for the Fortran COMPLEX types follow:
COMPLEX4 STRUCT 4 real REAL4 0 imag REAL4 0 COMPLEX4 ENDS COMPLEX8 STRUCT 8 real REAL8 0 imag REAL8 0 COMPLEX8 ENDS
A Fortran LOGICAL(2) is stored as a 2-byte indicator value (0=false, and the /fpscomp:[no]logicals compiler option determines how true values are handled). A Fortran LOGICAL(4) is stored as a 4-byte indicator value, and LOGICAL(1) is stored as a single byte. The type LOGICAL is the same as LOGICAL(4), which is equivalent to type int in C. You can use a variable of type LOGICAL in an argument list, module, common block, or global variable in Fortran and type int in C for the same argument. Type LOGICAL(4) is recommended instead of the shorter variants for use in common blocks. The Visual C++ class type has the same layout as the corresponding C struct type, unless the class defines virtual functions or has base classes. Classes that lack those features can be passed in the same way as C structures.
Page 31 of 55
contiguous array for non-contiguous array slices, which is like passing the first element of an array. When a Fortran 95/90 array pointer or array is passed to another language, either its descriptor or its base address can be passed. The following shows how allocatable arrays and Fortran 95/90 array pointers are passed with different attributes in effect:
l
If the property of the array pointer or array is none, it is passed by descriptor, regardless of the property of the passing procedure (None; C; STDCALL; C, REFERENCE; or STDCALL, REFERENCE). If the property of the array pointer or array is VALUE, an error is returned, regardless of the property of the passing procedure. If the property of the array pointer or array is REFERENCE, it is passed by descriptor, regardless of the property of the passing procedure.
Note that the VALUE option cannot be used with descriptor-based arrays. When you pass a Fortran array pointer or an array by descriptor to a nonFortran routine, that routine needs to know how to interpret the descriptor. Part of the descriptor is a pointer to address space, as a C pointer, and part of it is a description of the pointer or array properties, such as its rank, stride, and bounds. For information about the Visual Fortran array descriptor format, see Handling Arrays and Visual Fortran Array Descriptors. Fortran 95/90 pointers that point to scalar data contain the address of the data and are not passed by descriptor. For information about performance implications of passing different types of array arguments, see Use Arrays Efficiently.
The argument should be declared in the non-Fortran routine as a pointer of the appropriate data type. The argument passed from the Fortran routine should be the Compaq
Programming with Mixed Languages Fortran pointer name, not the pointer-based variable name. For example, on ia32 systems:
! Fortran main program. INTERFACE SUBROUTINE Ptr_Sub (p) !DEC$ ATTRIBUTES C, ALIAS:'_Ptr_Sub' :: Ptr_Sub INTEGER p END SUBROUTINE Ptr_Sub END INTERFACE REAL A(10), VAR(10) POINTER (p, VAR) ! VAR is the pointer-based ! variable, p is the int. p = LOC(A) CALL Ptr_Sub (p) WRITE(*,*) 'A(4) = ', A(4) END ! //C subprogram void Ptr_Sub (float *p) { p[3] = 23.5; }
Page 32 of 55
On ia64 systems, the alias name for Ptr_Sub should not have a leading underscore, as follows:
!DEC$ ATTRIBUTES C, ALIAS:'Ptr_Sub' :: Ptr_Sub
When the main Fortran program and C function are built and executed, the following output appears:
A(4) = 23.50000
The argument should be declared in the non-Fortran routine as a pointer of the appropriate data type and passed as usual. The argument received by the Fortran routine should be declared as a Compaq Fortran pointer name, then the POINTER statement should associate it with a pointer-based variable of the appropriate data type (matching the data type of the passing routine). When inside the Fortran routine, use the pointer-based variable to set and access what the pointer points to.
Page 33 of 55
On ia64 systems, the alias name for Iptr_Sub should not have a leading underscore, as follows:
!DEC$ ATTRIBUTES C, ALIAS:'Iptr_Sub' :: Iptr_Sub
When the main C program and Fortran subroutine are built and executed, the following output appears if the STAT.DAT file contains 4:
a[3] = 4
Page 34 of 55
Option Base 1
Alternatively, in the array declaration in either language you can set the array lower bound to any integer in the range -32,768 to 32,767. For example:
' In Basic Declare Sub FORTARRAY Lib "fortarr.dll" (Barray as Single) DIM barray (1 to 3, 1 to 7) As Single Call FORTARRAY(barray (1,1)) ! In Fortran Subroutine FORTARRAY(arr) REAL arr(3,7)
In MASM, arrays are one-dimensional and array elements must be referenced byte-by-byte. The assembler stores elements of the array consecutively in memory, with the first address referenced by the array name. You then access each element relative to the first, skipping the total number of bytes of the previous elements. For example:
xarray REAL4 1.1, 2.2, 3.3, 4.4 ; initializes ; a four element array with ; each element 4 bytes
Referencing xarray in MASM refers to the first element, the element containing 1.1. To refer to the second element, you must refer to the element 4 bytes beyond the first with xarray[4] or xarray+4. Similarly:
yarray 256 DUP ; establishes a ; 256 byte buffer, no initialization zarray SWORD 100 DUP(0) ; establishes 100 ; two-byte elements, initialized to 0 BYTE
The value of the lower array bound is different. By default, Fortran indexes the first element of an array as 1. C and Visual C++ index it as 0. Fortran subscripts should therefore be one higher. (Fortran also provides the option of specifying another integer lower bound.) In arrays of more than one dimension, Fortran varies the left-most index the fastest, while C varies the right-most index the fastest. These are sometimes called column-major order and row-major order, respectively.
Page 35 of 55
The order of indexing extends to any number of dimensions you declare. For example, the C declaration:
int arr1[2][10][15][20];
The constants used in a C array declaration represent extents, not upper bounds as they do in other languages. Therefore, the last element in the C array declared as int arr[5][5] is arr[4][4], not arr[5][5]. The following table shows equivalencies for array declarations. Equivalent Array Declarations for Different Languages Language Fortran Array Declaration DIMENSION x(i, k) -ortype x(i, k) DIM x(i, k) As type type x[ k ] [ i ] Array Reference from Fortran x(i, k)
Visual Fortran Array Descriptor Format For cases where Fortran 95/90 needs to keep track of more than a pointer memory address, the Visual Fortran compiler uses an array descriptor, which stores the details of how an array is organized. When using an explicit interface (by association or procedure interface block), Visual Fortran will generate a descriptor for the following types of array arguments:
l l
Certain data structure arguments do not use a descriptor, even when an appropriate explicit interface is provided. For example, explicit-shape and
Page 36 of 55
assumed-size arrays do not use a descriptor. In contrast, array pointers and allocatable arrays use descriptors regardless of whether they are used as arguments. When calling between Visual Fortran and a non-Fortran language (such as C), using an implicit interface allows the array argument to be passed without a Visual Fortran descriptor (see Passing Array Arguments Efficiently). However, for cases where the called routine needs the information in the Visual Fortran descriptor, declare the routine with an explicit interface and specify the dummy array as either an assumed-shape array or with the pointer attribute. You can associate a Fortran 95/90 pointer with any piece of memory, organized in any way desired (so long as it is "rectangular" in terms of array bounds). You can also pass Fortran 95/90 pointers to other languages, such as C, and have the other language correctly interpret the descriptor to obtain the information it needs. However, using array descriptors can increase the opportunity for errors and is not portable:
l
If the descriptor is not defined correctly, the program might access the wrong memory address, possibly causing a General Protection Fault. Array descriptor formats are specific to each Fortran compiler. Code that uses array descriptors is not portable to other compilers or platforms. For example, the Visual Fortran array descriptor format (for Win32 systems) differs from the array descriptor format for Compaq Fortran on Compaq Tru64 UNIX and Compaq OpenVMS systems. The Visual Fortran array descriptor format is the same format used by Microsoft Fortran PowerStation. The array descriptor format might change in the future.
The components of the current Visual Fortran array descriptor on ia32 systems follow:
l
The first longword (bytes 0 to 3) contains the base address. The base address plus the offset defines the first memory location (start) of the array. The second longword (bytes 4 to 7) contains the size of a single element of the array. The third longword (bytes 8 to 11) contains the offset. The offset is added to the base address to define the start of the array. The fourth longword (bytes 12 to 15) contains the low-order bit set if the array has been defined (storage allocated). Other bits may also be set by the compiler within this longword, for example, to indicate a contiguous array. The fifth longword (bytes 16 to 19) contains the number of dimensions (rank) of the array.
Page 37 of 55
The remaining longwords (bytes 20 up to 103) contain information about each dimension (up to seven). Each dimension is described by three additional longwords: The number of elements (extent) The distance between the starting address of two successive elements, in bytes. The lower bound
An array of rank one would require three additional longwords for a total of in eight longwords (5 + 3*1) and end at byte 31. An array of rank seven would be described in a total of 26 longwords (5 + 3*7) and end at byte 103. For example, consider the following declaration:
integer,target :: a(10,10) integer,pointer :: p(:,:) p => a(9:1:-2,1:9:3) call f(p) . . .
The descriptor for actual argument p would contain the following values:
l
l l
l l l
The first longword (bytes 0 to 3) contain the base address (assigned at run-time). The second longword (bytes 4 to 7) is set to 4 (size of a single element). The third longword (bytes 8 to 11) contain the offset (assigned at runtime). The fourth longword (bytes 12 to 15) contains 1 (low bit is set). The fifth longword (bytes 16 to 19) contains 2 (rank). The sixth, seventh, and eighth longwords (bytes 20 to 31) contain information for the first dimension, as follows: 5 (extent) -8 (distance between elements) 1 (the lower bound) For the second dimension, the ninth, tenth, and eleventh longwords (bytes 32 to 43) contain: 3 (extent) 120 (distance between elements) 1 (the lower bound) Byte 43 is the last byte for this example.
For information about performance implications of passing different types of array arguments, see Passing Array Arguments Efficiently.
Page 38 of 55
By default, Visual Fortran passes a hidden length argument for strings. The hidden length argument consists of an unsigned 4-byte integer (ia32 systems) or unsigned 8-byte integer (ia64 systems), always passed by value, immediately following the address of the character string. You can alter the default way strings are passed by using attributes. The following table shows the effect of various attributes on passed strings. Effect of ATTRIBUTES Options on Character Strings Passed as Arguments Argument String Default Passed by reference, along with length C First character converted to INTEGER (4) and passed by value First character converted to INTEGER (4) and passed by value Passed by reference, no length STDCALL First character converted to INTEGER (4) and passed by value First character converted to INTEGER (4) and passed by value Passed by reference, no length C, REFERENCE Passed by reference, along with length STDCALL, REFERENCE Passed by reference, along with length
Error
Character strings without the VALUE or REFERENCE attribute that are passed to C or STDCALL routines are not passed by reference. Instead, only the first character is passed and it is passed by value. Character strings with the VALUE option passed to C or STDCALL routines are not passed by reference. Instead, only the value of the first character is
Page 39 of 55
passed. For string arguments with default ATTRIBUTES, ATTRIBUTES C, REFERENCE, or ATTRIBUTES STDCALL, REFERENCE: When /iface:mixed_str_len_arg is set, the length of the string is pushed (by value) on the stack immediately after the address of beginning of the string. When /iface:nomixed_str_len_arg is set, the length of the string pushed (by value) on the stack after all of the other arguments. For string arguments passed by reference with default ATTRIBUTES: When /iface:mixed_str_len_arg is set, the length of the string is pushed (by value) on the stack immediately after the address of beginning of the string. When /iface:nomixed_str_len_arg is set, the length of the string available to the called procedure.
the is
the is not
Since all strings in C are pointers, C expects strings to be passed by reference, without a string length. In addition, C strings are null-terminated while Fortran strings are not. There are two basic ways to pass strings between Fortran and C: convert Fortran strings to C strings, or write C routines to accept Fortran strings. To convert a Fortran string to C, choose a combination of attributes that passes the string by reference without length, and null terminate your strings. For example, on ia32 systems:
INTERFACE SUBROUTINE Pass_Str (string) !DEC$ ATTRIBUTES C, ALIAS:'_Pass_Str' :: Pass_Str CHARACTER*(*) string !DEC$ ATTRIBUTES REFERENCE :: string END SUBROUTINE END INTERFACE CHARACTER(40) forstring DATA forstring /'This is a null-terminated string.'C/
On ia64 systems, the first !DEC$ ATTRIBUTES line would omit the leading underscore and be as follows:
!DEC$ ATTRIBUTES C, ALIAS:'Pass_Str' :: Pass_Str
This example shows the extension of using the null-terminator for the string in the Fortran DATA statement (see C Strings):
DATA forstring /'This is a null-terminated string.'C/
To get your C routines to accept Fortran strings, C must account for the length
Programming with Mixed Languages argument passed along with the string address. For example:
! Fortran code INTERFACE SUBROUTINE Pass_Str (string) CHARACTER*(*) string END INTERFACE
Page 40 of 55
This interface handles the hidden-length argument, but you must still reconcile C strings that are null-terminated and Fortran strings that are not. In addition, if the data assigned to the Fortran string is less than the declared length, the Fortran string will be blank padded. Rather than trying to handle these string differences in your C routines, the best approach in Fortran/C mixed programming is to adopt C string behavior whenever possible. Another good reason for using C strings is that Win32 APIs and most C library functions expect null-terminated strings. Fortran functions that return a character string using the syntax CHARACTER*(*) place a hidden string argument and the address of the string at the beginning of the argument list. C functions that implement such a Fortran function call must declare this hidden string argument explicitly and use it to return a value. The C return type should be void. However, you are more likely to avoid errors by not using characterstring return functions. Use subroutines or place the strings into modules or global variables whenever possible. Visual Basic strings must be passed by value to Fortran. Visual Basic strings are actually stored as structures containing length and location information. Passing by value dereferences the structure and passes just the string location, as Fortran expects. For example:
! In Basic Declare Sub forstr Lib "forstr.dll" (ByVal Bstring as String) DIM bstring As String * 40 Fixed-length string CALL forstr(bstring) ! End Basic code ! In Fortran SUBROUTINE forstr(s) !DEC$ ATTRIBUTES STDCALL :: forstr !DEC$ ATTRIBUTES REFERENCE :: s CHARACTER(40) s s = 'Hello, Visual Basic!' END
Page 41 of 55
The Fortran directive !DEC$ ATTRIBUTES STDCALL and the ATTRIBUTES REFERENCE property on variable argument s together inform Fortran not to expect the hidden length arguments to be passed from the Visual Basic calling program. The name in the Visual Basic program is specified as lowercase since STDCALL makes the Fortran name lowercase. MASM does not add either a string length or a null character to strings by default. To append the string length, use the syntax: lenstring BYTE "String with length", LENGTHOF lenstring To add a null character, append it by hand to the string:
nullstring BYTE "Null-terminated string", 0
In the Fortran code above, the SEQUENCE statement preserves the storage order of the derived-type definition. C Code:
/* C code accessing D1 and D2 */ extern struct { struct { float a; int b; char info[30]; struct { float real, imag; } cx; char moreinfo[80]; } d1, d2; } T_BLOCK;
Page 42 of 55
The Fortran (DF) compiler accepts an object file for the main program written in C and compiled by the C compiler. The DF compiler compiles the .F90 file and then has the linker create an executable file under the name CMAIN.EXE using the two object files. Either compiler can do the linking, regardless of which language the main program is written in; however, if you use the DF compiler first, you must include DFOR.LIB with the C compiler, and you might experience some difficulty with the version of LIBC.LIB used by the C compiler. For these reasons, you may prefer to use the C compiler first or get your project settings for both Fortran and C to agree on the default C library to link against making sure that your application links against one and only one copy of the C library. When using the visual development environment to build your application, Fortran uses default libraries depending on the information specified in the Fortran tab in the Project menu, Settings item (Project Settings dialog box). You can also specify linker settings with the Linker tab in the Project Settings dialog box. In the Fortran tab, within the Libraries category, the following options determine
Page 43 of 55
Use Fortran Run-time Libraries (see Types of Projects) Use Multi-threaded Libraries (see /[no]threads) Use C Debug Libraries (see /[no]dbglibs)
The combinations of these options use the following libraries: Static or DLL Project? Static Static Static Static DLL DLL DLL DLL Use MultiTheaded Libraries? No No Yes Yes No No Yes Yes Use C Debug Libraries? No Yes No Yes No Yes No Yes Fortran Link Library Used dfor.lib dfor.lib dformt.lib dformt.lib dfordll.lib (dforrt.dll) dfordlld.lib (dforrtd.dll) dformd.lib (dformd.dll) dformdd.lib (dformdd.dll) C Link Library Used libc.lib libcd.lib libcmt.lib libcmtd.lib msvcrt.lib (msvcrt.dll) msvcrtd.lib (msvcrtd.dll) msvcrt.lib (msvcrt.dll) msvcrtd.lib (msvcrtd.dll)
For example, if you select the Single-threaded item in the Use Run-Time Libraries list in the Libraries category of the Fortran tab, this specifies that static (/libs:static), single-threaded (/nothreads), and non-debug (/nodbglibs) libraries will be linked against, namely Fortran library dfor.lib and C/C++ library libc.lib. If you select Debug Multi-threaded DLL in the Use Run-Time Libraries list in the Libraries category of the Fortran tab, this specifies that DLL (/libs:DLL), multithreaded (/threads), and debug (/dbglibs) libraries will be linked against, namely Fortran import library dformdd.lib and its DLL library dformdd.dll and C/C++ import library msvcrtd.lib and its DLL library msvcrtd.dll. The way Visual C++ chooses libraries is also based upon the Project menu Settings item, but within the C/C++ tab. In the Code Generation category, the "Use run-time library" item lists the following C libraries:
Page 44 of 55
Menu Item Selected Single-threaded Multithreaded Multithreaded DLL Debug Singlethreaded Debug Multithreaded Debug Multithreaded DLL
CL Option or Project Type Enabled /ML /MT /MD /MLd /MTd /MDd
Default Library Specified in Object File libc.lib libcmt.lib msvcrt.lib (msvcrt.dll) libcd.lib libcmtd.lib msvcrtd.lib (msvcrtd.dll)
If you are using Microsoft Visual C/C++, the Microsoft visual development environment can create mixed Fortran/C applications transparently, with no special directives or steps on your part. You can edit and browse your C and Fortran programs with appropriate syntax coloring for the language. You can add C source files to your Fortran project or Fortran source files to a C project, and they will be compiled and linked automatically. When you debug a mixed Visual C/Fortran application, the debugger will adjust to the code type as it steps through: the C or Fortran expression evaluator will be selected automatically based on the code being debugged, and the stack window will show Fortran data types for Fortran procedures and C data types for C procedures. When printing from Visual C++ programs while calling Fortran subprograms that also print, the output may not appear in the order you expect. In Visual C++, the output buffer contents are not written immediately, but written when the buffer is full, the I/O stream is closed or the program terminates normally. The buffer is said to be "flushed" when this occurs. To make sure interleaving Visual C++ and Fortran program units print in the order expected, you can explicitly flush the Visual C++ buffers after an output command with the flushall, fflush, fclose, setbuf, or setvbuf Visual C++ library calls. Multithreaded applications should have full multithread support, so if you use DFORMT.LIB, be sure LIBCMT.LIB is specified as a default library.
Page 45 of 55
Visual Basic for the user-interface features Visual Fortran for computation
In Fortran/Visual Basic programs, the Visual Basic must be 32-bit (at least Version 5.0). You can also use the Visual Basic for Applications (VBA) included with Microsoft Excel to call Fortran subprograms. The Visual Basic development environment is separate from the Visual Fortran Version 6 development environment. However, the two languages can coexist in the same final application. The usual case is to call Fortran subprograms from Visual Basic. Because Visual Basic subprograms are interpreted and not compiled, they cannot be called directly from compiled language programs like Fortran. Instead, Visual Basic creates OLE objects that export properties and routines. It is also possible for Visual Basic to pass the address of its procedures to a Fortran program, to be called later by that Fortran program as callbacks. Refer to the Visual Fortran Sample in the ...\DF98\SAMPLES\MIXLANG\VB\CALLBACK folder for an example of Visual Fortran calling Visual Basic callback routines. When calling a Fortran subprogram from Visual Basic, you need to: 1. Create the Visual Fortran subprogram as a Fortran DLL project 2. Reference the DLL from Visual Basic with a Declare Sub or Declare Func statement The following sections discuss Visual Basic calling a Fortran DLL subprogram:
l l l
Calling Visual Fortran from Visual Basic Visual Basic Debugging Considerations Examples of Fortran/Visual Basic Programs
Visual Basic uses the STDCALL standard of argument passing with one exception: it does not append the "@n" count to the name on ia32 systems.
Page 46 of 55
Within Visual Basic, you declare the name of the routine that will be called and the arguments to be passed to it. Scalar numeric arguments can be directly passed from Basic and directly used by Fortran, but strings, arrays, and types require some extra handling.
Declaring the Fortran Routine in Visual Basic Exporting the Routine from the Fortran DLL Data Type Considerations
Declaring the Fortran Routine in Visual Basic When you declare a Fortran routine in Basic the routine name is exported in exactly the same case as you declared it. Optionally, you can specify a directory path to the Fortran DLL in the declaration. The following Visual Basic example declares a Fortran subroutine named FortranCall that takes two single-precision arguments:
Declare Sub FortranCall Lib "d:\MyProjects\FCall.dll" (A1 as Single, A2 as Single)
The following example declares a Fortran function named FortranFunc that takes two integer (32-bit) arguments and returns a single precision value:
Declare Function FortranFunc Lib "d:\MyProject\FFun.dll" (A1 as Long, A2 as Long) As Single
Exporting the Routine from the Fortran DLL When you create the Fortran DLL, you need to add two additional attributes to the function or subroutine declaration so that it can be accessed from outside the DLL. The Fortran routine name must be exported from the DLL, and it must be aliased to match exactly the name expected by Basic. The ATTRIBUTES (cDEC$ ATTRIBUTES compiler directive) and ALIAS declarations export the name and alias, as shown below for a Fortran subroutine:
SUBROUTINE FortranCall !DEC$ ATTRIBUTES DLLEXPORT :: FortranCall ! This exports the name !DEC$ ATTRIBUTES ALIAS : "FortranCall" :: FortranCall !This sets it
This is also true for Fortran functions, which must be exported and aliased to match exactly the name expected by Basic:
REAL Function FortranFunc !DEC$ ATTRIBUTES DLLEXPORT :: FortranFunc
Page 47 of 55
Data Type Considerations When you pass data between Visual Basic and Visual Fortran, you need to keep in mind the calling standard used by the two languages, the size of the data, and the on-disk format of the data. The following table summaries the calling considerations for the data types: Data Type Integer Calling Considerations By default, both Visual Basic and Visual Fortran pass integers by reference. The default integer size in Basic is 2 bytes, equivalent to INTEGER(2) in Fortran. No extra action is required by either language to access integer arguments. By default, both Visual Basic and Visual Fortran pass single- and double-precision floating point numbers by reference. The size of a single-precision floating point number is 4 bytes in both languages. The size of a double-precision floating point number is 8 bytes in both languages. No extra action is required by either language to access floating point arguments. By default, both Basic and Fortran pass logical data by reference. The default logical size in Basic is two bytes, equivalent to LOGICAL (2) in Fortran. No extra action is required by either language to access logical arguments. By default, Basic passes strings in a structure called a BSTR. By default, Fortran passes strings in two arguments: the string's address and a hidden argument containing the string's length. These defaults can be easily overridden by making changes in the Basic declaration, and sometimes in the Fortran declaration too. Whenever you pass a string from Basic to Fortran, the passing mechanism should be declared as ByVal. See the String Passing Examples. Arrays By default, Basic arrays are 0-based. Fortran arrays are 1-based by default. When you declare a Basic array to be of size n, n+1 elements are allocated, including the 0th element. This can be overridden, as show in the Array Passing Examples. To pass arrays of numbers from Basic to Fortran, whether they are integer or floating-point numbers, pass the first element. In the
Floating point
Logical
Strings
Page 48 of 55
Fortran code declare the argument to be an array. Usually the Basic code will pass the number of elements in the array. To pass arrays of strings or types from Basic to Fortran requires the use of COM utilities to read the structures created by Basic. The arrays are passed as is from Basic, and then extracted in the Fortran code using SafeArrayxx utilities. For more detail, please refer to the Visual Fortran Samples found in the MixLang\VB\arrays and MixLang\VB\typearrays folders. Types Basic-declared types can be passed from Basic to Fortran. The Basic-declared type will also have to be declared in Fortran; care should be taken to keep the two structures the same. In Fortran, the TYPE should be declared to be packed, usually PACK:2. You should be aware of the default sizes of the TYPE elements in the two languages, and adjust the defaults accordingly. Note that strings in Basic-declared types are stored as Unicode, which can be declared in Fortran as an array of Integer*2. To access these strings from Fortran, use the Natural Language System (NLS) API calls. For information, see Using National Language Support Routines and the Visual Fortran Samples found in the MixLang\VB\typearrays folder. String Passing Examples The following example shows passing strings using the usual case, where the string will have varying length, and the length must also be passed to Fortran ByVal: Visual Basic code:
Declare Sub FortString1 Lib "forttest" (ByVal S1 as String, ByVal L1 as Long) Dim S1 as String * 12 Call FortString1(S1, Len(S1))
The following example shows shows passing strings where the length of the string will be constant and known by both the Basic and Fortran code, so you do not need to pass the length to Fortran, but you need to tell Fortran not to expect its length:
Page 49 of 55
Array Passing Examples The following example shows shows passing arrays: Visual Basic code:
Basic declaration: Declare Sub FortArray1 Lib "forttest" (A1 as Long, NumElem as long) Dim A1(1:3) as Long Call FortArray1(A1(1), 3)
Debugging the Fortran DLL Visual Basic Error 53: File not found: yy.dll Visual Basic Error 453: Can't find DLL entry point xx in yy.dll
Debugging the Fortran DLL In Visual Basic, create an executable of your project. Use the pull down menu item File-Make myproject.exe. Note where the executable was created. In the Visual Fortran development environment: 1. In the Project menu, click Settings 2. Click the Debug tab 3. In the box labelled "Executable for debug session," enter the full path and
Programming with Mixed Languages filename of the executable you created above.
Page 50 of 55
You can now use the full power of the visual development environment debugger to debug your DLL. If you are using VBA within Microsoft Excel, the steps are similar for debugging your DLL. Enter the full path and filename for Excel into the box labelled "Executable for debug session". Optionally, you can enter the name of the worksheet into the box labeled "Program arguments:". You can now use the visual development environment to debug your DLL. For a sample Fortran debugging session, see Debugging the Squares Example Program. Visual Basic Error 53: File not found: yy.dll If you get the error message "File not found: yy.dll" when you run the Visual Basic application, check the following:
l
If you built the DLL on one system and copied it to another, did you also copy the Fortran run-time DLLs? These files are DFORRT.DLL, DFORMD.DLL, and MSVCRT.DLL on the Visual Fortran CD-ROM. They can be freely distributed with your application (see Compaq Visual Fortran Installing and Getting Started).
Is the correct path to yy.dll specified in the Basic declaration? Carefully check the path and file name in the Visual Basic Declare statement.
Visual Basic Error 453: Can't find DLL entry point xx in yy.dll If you get the error message "Can't find DLL entry point xx in yy.dll," check the following:
l
Make sure the Fortran code has specified the ATTRIBUTES DLLEXPORT for the routine name (see the cDEC$ ATTRIBUTES compiler directive). Make sure the Fortran code has specified an ATTRIBUTES ALIAS name that exactly matches the name declared by the Basic code (see the cDEC$ ALIAS directive, which also can be used as an ATTRIBUTES option ALIAS). Make sure you are referencing the correct yy.dll.
For an example of calling a Fortran DLL from Visual Basic, see Examples of Fortran/Visual Basic Programs.
Page 51 of 55
The following brief code demonstrates the interface for a Fortran subroutine and function (free-form Fortran source): 1. In the visual development environment, create a new project of type Fortran Dynamic-Link Library. Name the project FCALL. 2. Create a new free-form source file (Project menu, Add to Project, New) for the project named FCALL.F90 with the following code:
! Fortran Code establishing subroutine ! Computes the MOD of R1 and 256.0 and stores the ! result in the argument NUM SUBROUTINE FortranCall (r1, num) ! Specify that the routine name is to be made available to callers of the ! DLL and that the external name should not have any prefix or suffix !DEC$ ATTRIBUTES DLLEXPORT :: FortranCall !DEC$ ATTRIBUTES ALIAS:'FortranCall' :: FortranCall REAL,INTENT(IN) :: r1 REAL,INTENT(OUT) :: num num = MOD (r1, 256.0) END SUBROUTINE ! Input argument
3. Build the Fortran DLL as described in Building and Using Dynamic-Link Libraries. 4. Start Visual Basic and create a new Standard EXE project: On the control toolbar, click the CommandButton icon and then, with the cursor over the form, draw out a button. In the button's Properties box, double-click on Caption and change the caption to "Do it!". Click the TextBox icon and, in the same fashion, draw a text box on the form. In its Properties box, find the Text property and change it to an empty string. 5. Double-click the Command button on the form - a code window will appear. Fill in the code so that it looks like this:
Private Sub Command1_Click() r1 = 456.78 Call FortranCall(r1, Num) Text1.Text = Str$(Num) End Sub
6. Select Project..Add Module and click Open to create a new module. Add the following code to the module:
Declare Sub FortranCall Lib"c:\MyProjects\Fcall\Debug\Fcall.dll" (r1 As Single, Num As Single)
Page 52 of 55
Replace the filename with the location of the Fortran DLL, if it is different. 7. Run the Basic program by pressing F5. Click the Do it! button. The Fortran routine will be called to compute the modulus, returning the result to the Basic code. The Basic code will then convert the result to a string and display it in the text box. Visual Basic, like Fortran, passes numeric values (such as integers and reals) by reference, so it is not necessary to change the passing mechanism on either side. The ALIAS attribute is required because Visual Basic, even though it uses the STDCALL calling mechanism, does not "decorate" routine names with the @n suffix. If the Fortran routine were also to be called by other Fortran code, it would be appropriate to use the Alias option on the Basic side to name it with the proper suffix.
Page 53 of 55
In the example, the Fortran call to MASM is power2(v,e), which is identical to a Fortran function call. Visual Fortran also provides sample mixed-language programs that show calls between Visual Fortran and MASM (see Examples of Fortran/MASM Programming). There are two differences between this mixed-language call and a call between two Fortran modules: 1. The subprogram power2(v,e) is implemented in MASM using standard MASM syntax. The PROTO declaration in MASM specifies that the procedure use the STDCALL calling convention. 2. The INTERFACE statement in the Fortran module specifies the STDCALL calling convention, so the Fortran program uses same convention that the MASM procedure specifies. This section covers the following topics:
l l l
Creating a MASM Procedure Fortran/MASM Alignment and Return Value Considerations Examples of Fortran/MASM Programming
Page 54 of 55
Your MASM procedure can return a value to your Fortran routine if you prototype it as a function. All return values of 4 bytes or less (except for floating-point values) are returned in the EAX register. Procedures that return floating-point values return their results on the floatingpoint processor stack. This is possible because there is always a coprocessor or emulator available for 32-bit compilers. To return REAL and COMPLEX floating-point values, records, arrays, and values larger than 4 bytes and return user-defined types larger than 8 bytes from assembly language to Fortran, you must use a special convention. Fortran creates space in the stack segment to hold the actual return value and passes an extra parameter as the last parameter pushed onto the stack. This extra parameter contains the address of the stack space that contains the return value. For user-defined types, values of 4 bytes or less are returned in EAX and values of 5 to 8 bytes are returned in EAX:EDX. In the assembly procedure, put the data for the return value at the location pointed to by the return value offset. Then copy the return-value offset (located at EBP+8 if you've created a stack frame in your assembly code) to EAX. This is necessary because the calling module expects EAX to point to the return value. The following table summarizes ways to return values. Summary of Ways to Return Values Type of Value to Return Integer, logical variable, or user-defined type of size 4 bytes or less Floating-point variable Variable of size more than 4 bytes (strings, complex values) or user-defined types more than 8 bytes User-defined structures between 5 and 8 bytes Method of Returning Value Return value in EAX register Return value on the FPU stack Return value on stack, address of value in EAX register Return value in EAX:EDX registers.
Page 55 of 55
Several Visual Fortran sample programs have been provided that illustrate Visual Fortran routines that call MASM procedures. These samples are located in subfolders of the ...\DF98\SAMPLES\MIXLANG\MASM folder and include simple, vecprod, and allocabl.
Page 1 of 17
Basic Concepts of Multithreading Writing a Multithread Program Compiling and Linking Multithread Programs Other Sources of Information
Page 2 of 17
program runs in the first thread. A Win32 thread consists of a stack, the state of the CPU registers, a security context, and an entry in the execution list of the system scheduler. Each thread shares all of the process's resources. A process consists of one or more threads and the code, data, and other resources of a program in memory. Typical program resources are open files, semaphores (a method of interthread communication), and dynamically allocated memory. A program executes when the system scheduler gives one of its threads execution control. The scheduler determines which threads should run and when they should run. Threads of lower priority might need to wait while higher priority threads complete their tasks. On multiprocessor machines, the scheduler can move individual threads to different processors to balance the CPU load. Because threads require less system overhead and are easier to create than an entire process, they are useful for time- or resource-intensive operations that can be performed concurrently with other tasks. Threads can be used for operations such as background printing, monitoring a device for input, or backing up data while it is being edited. When threads, processes, files, and communications devices are opened, the function that creates them returns a handle. Each handle has an associated Access Control List (ACL) that is used to check the security credentials of the process. Processes and threads can inherit a handle or give one away using functions described in this section. Objects and handles regulate access to system resources. For more information on handles and security, see the Win32 Application Programming Interface reference (such as the Platform SDK online title). All threads in a process execute independently of one another. Unless you take special steps to make them communicate with each other, each thread operates while completely unaware of the existence of other threads in a process. Threads sharing common resources must coordinate their work by using semaphores or another method of interthread communication. For more information on interthread communication, see Sharing Resources.
Background tasks such as data calculations, database queries, and input gathering, which do not directly involve window management or user interface. Operations that are independent from one another that can benefit from concurrent processing. Asynchronous tasks such as polling on a serial port.
Page 3 of 17
If your application contains tasks that require a private address space and private resources, you can protect them from the activities of other threads by creating multiple processes rather than multiple threads. See Working with Multiple Processes. The sections that follow discuss the steps you need to consider in creating a multithread application:
l l l l l l l l l
Modules for Multithread Programs Starting and Stopping Threads Thread Routine Format Sharing Resources Thread Local Storage (TLS) Synchronizing Threads Handling Errors in Multithread Programs Working with Multiple Processes Table of Multithread Routines
Page 4 of 17
One single call to the CreateThread function creates a thread, specifies security attributes and memory stack size, and names the routine for the thread to run. Windows allocates memory for the thread stack in the virtual address space of the application that contains the thread. Once a thread has finished processing, the CloseHandle routine frees the resources used by the thread. For more information:
l l l
Starting Threads The function CreateThread creates a new thread. Its return value is an INTEGER (4) thread handle, used in communicating to the thread and when closing it. The syntax for this function is: CreateThread (security, stack, thread_func, argument, flags, thread_id) All arguments are INTEGER(4) variables except for thread_func, which names the routine for CreateThread to run. Minimum requirements for thread_func are discussed in Thread Routine Format. The arguments are as follows: security This argument uses the SECURITY_ATTRIBUTES type, defined in DFMT.F90. If security is zero, the thread has the default security attributes of the parent process. For more information about setting security attributes for processes and threads, see the Platform SDK online reference. Defines the stack size of the new thread. All of an application's default stack space is allocated to the first thread of execution. As a result, you must specify how much memory to allocate for a separate stack for each additional thread your program needs. The CreateThread call allows you to specify the value for the stack size on each thread you create. A value of zero indicates the stack has the same size as the application's primary thread. The size of the stack is increased dynamically, if necessary, up to a limit of 1 MB. The starting address for the thread function. An optional argument for thread_func. Your program defines this parameter and how it is used.
stack
thread_func argument
Page 5 of 17
This argument lets you create a thread that will not begin processing until you signal it. The flags argument can take either of two values: 0 or CREATE_SUSPENDED. If you specify 0, the thread is created and runs immediately after creation. If you specify CREATE_SUSPENDED, the thread is created, but does not run until you call the ResumeThread function. This argument is returned by CreateThread. It is a unique identifier for the thread, which you can use when calling other multithread routines. While the thread is running, no other thread has the same identifier. However, the operating system may use the identifier again for other threads once this one has completed. A thread can be referred to by its handle as well as its unique thread identifier. Synchronization functions such as WaitForSingleObject and WaitForMultipleObjects take the thread handle as an argument.
thread_id
Stopping Threads The ExitThread routine allows a thread to stop its own execution. The syntax is: CALL EXITTHREAD ( [ Termination Status ] ) Termination status may be queried by another thread. A termination status of 0 indicates normal termination. You can assign other termination status values and their meaning in your program. When the called thread is no longer needed, the calling thread needs to close the handle for the thread. Use the CloseHandle routine to free memory used by the thread. A thread object is not deleted until the last thread handle is closed. It is possible for more than one handle to be open to a thread: for example, if a program creates two threads, one of which waits for information from the other. In this case, two handles are open to the first thread: one from the thread requesting information, the other from the thread that created it. All handles are closed implicitly when the enclosing process terminates. The TerminateThread routine allows one thread to terminate another, if the security attributes are set appropriately for both threads. DLLs attached to the thread are not notified that the thread is terminating, and its initial stack is not deallocated. Use Terminate Thread for emergencies only. Other Thread Support Functions
Page 6 of 17
Scheduling thread priorities is supported through the functions GetThreadPriority and SetThreadPriority. Use the priority class of a thread to differentiate between applications that are time critical and those that have normal or below normal scheduling requirements. If you need to manipulate priorities, be very careful not to give a thread too high a priority, or it can consume all of the available CPU time. A thread with a base priority level above 11 interferes with the normal operation of the operating system. Using REALTIME_PRIORITY_CLASS may cause disk caches to not flush, hang the mouse, and so on. When communicating with other threads, a thread uses a pseudohandle to refer to itself. A pseudohandle is a special constant that is interpreted as the current thread handle. Pseudohandles are only valid for the calling thread; they cannot be inherited by other threads. The GetCurrentThread function returns a pseudohandle for the current thread. The calling thread can use this handle to specify itself whenever a thread handle is required. Pseudohandles are not inherited. To get the thread's identifier, use the GetCurrentThreadId function. The identifier uniquely identifies the thread in the system until it terminates. You can use the identifier to specify the thread itself whenever an identifier is required. Use GetExitCodeThread to find out if a thread is still active, or if it is not, to find its exit status. Call GetLastError for more detailed information on the exit status. If one routine depends on a task being performed by a different thread, use the wait functions described in Synchronizing Threads instead of GetExitCodeThread.
Page 7 of 17
The arguments arg or arg2 are passed to the function or subroutine when the main program calls CreateThread, as the fourth argument. The arguments arg or arg2 are passed by value. Threads automatically terminate when the function or subroutine terminates.
Sharing Resources
Each thread has its own stack and its own copy of the CPU registers. Other resources, such as files, units, static data, and heap memory, are shared by all threads in the process. Threads using these common resources must coordinate their work. There are several ways to synchronize resources:
l
Critical section A block of code that accesses a non-shareable resource. Critical sections are typically used to restrict access to data or code that can only be used by one thread at a time within a process (for example, modification of shared data in a common block). MUTual EXclusion object (Mutex) A mechanism that allows only one thread at a time to access a resource. Mutexes are typically used to restrict access to a system resource that can only be used by one thread at a time (for example, a printer), or when sharing might produce unpredictable results. Semaphore A counter that regulates the number of threads that can use a resource. Semaphores are typically used to control access to a specified number of identical resources. Event An event object announces that an event has happened to one or more threads.
The state of each of these objects is either signaled or not-signaled. A signaled state indicates a resource is available for a process or thread to use it. A notsignaled state indicates the resource is in use. The routines described in the following sections manage the creation, initialization, and termination of resource sharing mechanisms. Some of them change the state to signaled from not-signaled. The routines WaitForSingleObject and WaitForMultipleObjects also change the signal status of an object. For information on these functions, see Synchronizing Threads. This section also contains information about:
l l
For resources about coordinating and synchronizing Win32 threads, see Other Sources of Information.
Page 8 of 17
Critical Sections Before you can synchronize threads with a critical section, you must initialize it by calling InitializeCriticalSection. Call EnterCriticalSection when beginning to process the global variable, and LeaveCriticalSection when the application is finished with it. Both EnterCriticalSection and LeaveCriticalSection can be called several times within an application. For Multithreaded Visual Fortran Samples that use Critical Sections, see PEEKAPP.F90 or PEEKAPP3.F90. Mutexes CreateMutex creates a mutex object. It returns an error if the mutex already exists (one by the same name was created by another process or thread). Call GetLastError after calling CreateMutex to look for the error status ERROR_ALREADY_EXISTS. You can also use the OpenMutex function to determine whether or not a named mutex object exists. When called, OpenMutex returns the object's handle if it exists, or null if a mutex with the specified name is not found. Using OpenMutex does not change a mutex object to a signaled state; this is accomplished by one of the wait routines described in Synchronizing Threads. ReleaseMutex changes a mutex from the not-signaled state to the signaled state. This function only has an effect if the thread calling it also owns the mutex. When the mutex is in a signaled state, any thread waiting for it can acquire it and begin executing. Semaphores Functions for handling semaphores are nearly identical to functions that manage mutexes. CreateSemaphore creates a semaphore, specifying an initial as well as a maximum count for the number of threads that can access the resource. OpenSemaphore, like OpenMutex, returns the handle of the named semaphore object, if it exists. The handle can then be used in any function that requires it (such as one of the wait functions described in Synchronizing Threads). Calling OpenSemaphore does not reduce a resource's available count; this is accomplished by the function waiting for the resource. Use ReleaseSemaphore to increase the available count for a resource by a specified amount. You can call this function when the thread is finished with the resource. Another possible use is to call CreateSemaphore, specifying an initial count of zero to protect the resource from access during an initialization process. When the application has finished its initialization, call ReleaseSemaphore to increase the resource's count to its maximum. Events
Page 9 of 17
Event objects can trigger execution of other threads. You can use events if one thread provides data to several other threads. An event object is created by the CreateEvent function. The creating thread specifies the initial state of the object and whether it is a manual-reset or auto-reset event. A manual-reset event is one whose state remains signaled until it is explicitly reset by a call to ResetEvent. An auto-reset event is automatically reset by the system when a single waiting thread is released. Use either SetEvent or PulseEvent to set an event object's state to signaled. OpenEvent returns a handle to the event, which can be used in other function calls. ReleaseEvent releases ownership of the event. Memory Use and Thread Stacks Because each thread has its own stack, you can avoid potential collisions over data items by using as little static data as possible. Design your program to use automatic stack variables for all data that can be private to a thread. All the variables declared in a multithread routine are by default static and shared among the threads. If you do not want one thread to overwrite a variable used by another, you can do one of the following:
l l
Declare the variable as AUTOMATIC. Create a vector of variable values, one for each thread, so that the variable values for different threads are in different storage locations. (You can use the single integer parameter passed by CreateThread as an index to identify the thread.) Use Thread Local Storage (TLS).
Variables declared as automatic are placed on the stack, which is part of the thread context saved with the thread. Automatic variables within procedures are discarded when the procedure completes execution. I/O Operations Although files and units are shared between threads, you may not need to coordinate the use of these shared resources by threads. Fortran treats each input/output statement as an atomic operation. If two separate threads try to write to the same unit and one thread's output operation has started, the operation will complete before the other thread's output operation can begin. The operating system does not impose an ordering on threads' access to units or files. For example, the non-determinate nature of multithread applications can cause records in a sequential file to be written in a different order on each execution of the application as each thread writes to the file. Direct access files might be a better choice than sequential files in such a case. If you cannot use direct access files, use mutexes to impose an ordering constraint on input or
Page 10 of 17
Certain restrictions apply to blocking functions for input procedures in QuickWin programs. For details on these restrictions, see Using QuickWin.
Synchronizing Threads
The routines WaitForSingleObject and WaitForMultipleObjects enable threads to wait for a variety of different occurrences, such as thread completion or signals from other threads. They enable threads and processes to wait efficiently, consuming no CPU resources, either indefinitely or until a specified timeout interval has elapsed. WaitForSingleObject takes an object handle as the first parameter and does not return until the object referenced by the handle either reaches a signaled state or until a specified timeout value elapses. The syntax is: WaitResult = WaitForSingleObject (ObjectHandle, [ Timeout ] ) If you are using a timeout, specify the value in milliseconds as the second parameter. The value WAIT_INFINITE represents an infinite timeout, in which case the function waits until ObjectHandle completes. WaitForMultipleObjects is similar, except that its second parameter is an array of Windows object handles. Specify the number of handles to wait for in the first parameter. This can be less than the total number of threads created, and its maximum is 64. The function can either wait until all events have completed, or resume as soon as any one of the objects completes. Deadlocks occur when a thread waits for objects that never become available. Use the timeout parameter when there is a chance that the thread you are
Page 11 of 17
waiting for may never terminate. See Detecting Deadlocks in Multithreaded Win32 Applications," by Ruediger Asche, in the Microsoft Systems Journal, vol. 8, for a discussion of how to find and avoid potential resource collisions. Suspending and Resuming Threads You can use SuspendThread to stop a thread from executing. SuspendThread is not particularly useful for synchronization because it does not control the point in the code at which the thread's execution is suspended. However, you could suspend a thread if you need to confirm a user's input that would terminate the work of the thread. If confirmed, the thread is terminated; otherwise, it resumes. If a thread is created in a suspended state, it does not begin to run until ResumeThread is called with a handle to the suspended thread. This can be useful for initializing the thread's state before it begins to run. Suspending a thread at creation can be useful for one-time synchronization, because ResumeThread ensures that the suspended thread will resume running at the starting point of its code.
The system can create and execute threads more quickly than it can create processes, since the code for threads has already been mapped into the address space of the process, while the code for a new process must be
Page 12 of 17
loaded. All threads of a process share the same address space and can access the process's global variables, which can simplify communications between threads. All threads of a process can use open handles to resources such as files and pipes.
If you want to create an independent process that runs concurrently with the current one, use CreateProcess. CreateProcess returns a 32-bit process identifier that is valid until the process terminates. ExitProcess stops the process and notifies all DLLs the process is terminating. Different processes can share mutexes, events, and semaphores (but not critical sections). Processes can also optionally inherit handles from the process that created them (see online help for CreateProcess). You can obtain information about the current process by calling GetCurrentProcess (returns a pseudohandle to its own process), and GetCurrentProcessId (returns the process identifier). The value returned by these functions can be used in calls to communicate with other processes. GetExitCodeProcess returns the exit code of a process, or an indication that it is still running. The OpenProcess function opens a handle to a process specified by its process identifier. OpenProcess allows you to specify the handle's access rights and inheritability. A process terminates whenever one of the following occurs:
l l l l
Any thread of the process calls ExitProcess The primary thread of the process returns The last thread of the process terminates TerminateProcess is called with a handle to the process
ExitProcess is the preferred way to terminate a process because it notifies all attached DLLs of the termination, and ensures that all threads of the process terminate. DLLs are not notified after a call to TerminateProcess.
Page 13 of 17
Routine CloseHandle CreateEvent CreateMutex CreateProcess CreateSemaphore CreateThread DeleteCriticalSection DuplicateHandle EnterCriticalSection ExitProcess ExitThread GetCurrentProcess GetCurrentProcessId GetCurrentThread GetCurrentThreadId GetExitCodeProcess GetExitCodeThread GetLastError GetPriorityClass GetThreadPriority InitializeCriticalSection
Description Closes an open object handle. Creates a named or unnamed event object. Creates a named or unnamed mutex object. Creates a new process and its primary thread. Creates a named or unnamed semaphore object. Creates a thread to execute within the address space of the calling process. Releases all resources used by an unowned critical section object. Duplicates an object handle. Waits for ownership of the specified critical section object. Ends a process and all its threads. Ends a thread. Returns a pseudohandle for the current process. Returns the process identifier of the calling process. Returns a pseudohandle for the current thread. Returns the thread identifier of the calling thread. Retrieves the termination status of the specified process. Retrieves the termination status of the specified thread. Returns the calling thread's last-error code value. Returns the priority class for the specified process. Returns the priority value for the specified thread. Initializes a critical section object.
Page 14 of 17
Releases ownership of the specified critical section object. Returns a handle of an existing named event object. Returns a handle of an existing named mutex object. Returns a handle of an existing process object. Returns a handle of an existing named semaphore object. As a single operation, sets (to signaled) and then resets the state of the specified event object after releasing the appropriate number of waiting threads. Releases ownership of the specified mutex object. Increases the count of the specified semaphore object by a specified amount. Sets the state of the specified event object to nonsignaled. Decrements a thread's suspend count. When the suspend count is zero, execution of the thread resumes. Sets the state of the specified event object to signaled. Sets the last-error code for the calling thread. Sets the priority class for the specified process. Sets the priority value for the specified thread. Suspends the specified thread. Terminates the specified process and all of its threads. Terminates a thread. Allocates a thread local storage (TLS) index. Releases a thread local storage (TLS) index, making it available for reuse.
Page 15 of 17
Retrieves the value in the calling thread's thread local storage (TLS) slot for a specified TLS index. Stores a value in the calling thread's thread local storage (TLS) slot for a specified TLS index. Returns either any one or all of the specified objects are in the signaled state or when the time-out interval elapses. Returns when the specified object is in the signaled state or the time-out interval elapses.
WaitForSingleObject
If a function mentioned in this section is not listed in the preceding table, it is only available through the USE DFWIN statement.
Page 16 of 17
5. Create the executable file by choosing Build All from the Build menu. The following steps describe how to compile and link the sample multithread program from the command line. To compile and link the sample multithread program from the command line: 1. Make sure the library files directory is specified in your LIB environment variable. 2. Compile and link the program with the DF command-line option /threads. For example:
DF /threads MYTHREAD.F90
The /threads compiler option (automatically set when you specify a multithread application in the visual development environment) tells the linker to use DFORMT.LIB as a default library. To compile and link the THREADS.F90 sample, the command is:
DF /winapp /threads THREADS.F90
The /threads compiler option causes the linker to search the multithread library; the /winapp requests a Fortran Windows application. Select the compiler options /libs=dll and /threads if you are using both multithread code and DLLs. You can use the /libs=dll and /threads options only with Fortran Console projects, not QuickWin applications.
The Microsoft Developer Network CD-ROM contains several articles on multithreading: "Multiple Threads in the User Interface," by Nancy Winnick Cluts, discusses the ramifications of adding multiple threads to the user interface. This article not only offers alternatives to multiple threads, but also covers window management and message loops for multithreading. "Multithreading for Rookies," by Ruediger R. Asche, focuses on practical applications of multithreading. "Detecting Deadlocks in Multithreaded Win32 Applications," by
Page 17 of 17
Ruediger R. Asche, presents deadlock detection techniques. A deadlock is a condition in which the application hangs because two or more threads are waiting for each other to release a shared resource before resuming execution. "Moving Unix Applications to Windows NT," provides an overview of Windows multithreading calls, contrasting them with Unix fork() calls.
The Microsoft Systems Journal is also a source of information on multithreading: "Coordinate Win32 threads using manual-reset and auto-reset events," by Jeffrey Richter. October 1993, v8 n10. "Synchronizing Win32 threads using critical sections, semaphores, and mutexes," by Jeffrey Richter. August 1993, v8 n8.
Data Representation
Page 1 of 9
Data Representation
Visual Fortran expects numeric data to be in native little endian order, in which the least-significant, right-most zero bit (bit 0) or byte has a lower address than the most-significant, left-most bit (or byte). For information on using nonnative big endian and Compaq VAX floating-point formats, see Converting Unformatted Numeric Data. The symbol :A in any figure specifies the address of the byte containing bit 0, which is the starting address of the represented data element. The following table lists the intrinsic data types used by Visual Fortran, the storage required, and valid ranges. For information on declarating Fortran intrinsic data types, see Type Declaration Statements. For example, the declaration INTEGER(4) is the same as INTEGER(KIND=4) and INTEGER*4. Compaq Fortran Data Types and Storage Data Type BYTE INTEGER(1) INTEGER Storage 1 byte (8 bits) See INTEGER (2), INTEGER (4), and INTEGER (8). 1 byte (8 bits) 2 bytes (16 bits) 4 bytes (32 bits) 8 bytes (64 bits) Description A BYTE declaration is a signed integer data type equivalent to INTEGER(1). Signed integer, either INTEGER(2), INTEGER(4), or INTEGER(8). The size is controlled by the /integer_size:nn compiler option. The default is /integer_size:32 (INTEGER(4)).
Signed integer value from -128 to 127. Signed integer value from -32,768 to 32,767. Signed integer value from -2,147,483,648 to 2,147,483,647. Signed integer value from 9,223,372,036,854,775,808 to 9,223,372,036,854,775,807.
Page 2 of 9 Single-precision real floating-point values in IEEE S_floating format ranging from 1.17549435E-38 to 3.40282347E38. Values between 1.17549429E-38 and 1.40129846E-45 are denormalized (subnormal). Double-precision real floating-point values in IEEE T_floating format ranging from 2.2250738585072013D-308 to 1.7976931348623158D308. Values between 2.2250738585072008D-308 and 4.94065645841246544D-324 are denormalized (subnormal). Single-precision complex floating-point values in a pair of IEEE S_floating format parts: real and imaginary. The real and imaginary parts range from 1.17549435E-38 to 3.40282347E38. Values between 1.17549429E-38 and 1.40129846E-45 are denormalized (subnormal). Double-precision complex floating-point values in a pair of IEEE T_floating format parts: real and imaginary. The real and imaginary parts each range from 2.2250738585072013D-308 to 1.7976931348623158D308. Values between 2.2250738585072008D-308 and 4.94065645841246544D-324 are denormalized (subnormal). Logical value, either LOGICAL(2), LOGICAL(4), or LOGICAL(8). The size is controlled by the /integer_size:nn compiler option. The default is /integer_size:32 (LOGICAL(4)).
COMPLEX(4) COMPLEX
LOGICAL
See LOGICAL (2), LOGICAL (4), and LOGICAL (8). 1 byte (8 bits) 2 bytes (16 bits) 4 bytes (32 bits)
Logical values .TRUE. or .FALSE. Logical values .TRUE. or .FALSE. Logical values .TRUE. or .FALSE.
Data Representation LOGICAL(8) CHARACTER 8 bytes (64 bits) 1 byte (8 bits) per character Logical values .TRUE. or .FALSE.
Page 3 of 9
Character data represented by character code convention. Character declarations can be in the form CHARACTER(LEN=n) or CHARACTER*n, where n is the number of bytes or n is (*) to indicate passed-length format. Hollerith constants.
HOLLERITH
In addition, you can define binary (bit) constants as explained in the Language Reference. The following sections discuss the intrinsic data types in more detail:
l l l l l
Integer Data Representations Logical Data Representations Native IEEE Floating-Point Representations Character Representation Hollerith Representation
INTEGER(KIND=1) Representation
INTEGER(1) values range from -128 to 127 and are stored in 1 byte, as shown
Page 4 of 9
INTEGER(KIND=2) Representation
INTEGER(2) values range from -32,768 to 32,767 and are stored in 2 contiguous bytes, as shown below: INTEGER(2) Data Representation
INTEGER(KIND=4) Representation
INTEGER(4) values range from -2,147,483,648 to 2,147,483,647 and are stored in 4 contiguous bytes, as shown below. INTEGER(4) Data Representation
INTEGER(KIND=8) Representation
Data Representation
Page 5 of 9
INTEGER(8) values range from -9,223,372,036,854,775,808 to 9,223,372,036,854,775,807 and are stored in 8 contiguous bytes, as shown below. INTEGER(8) Data Representation
Data Representation
Page 6 of 9
REAL(KIND=4) (REAL) Representation REAL(KIND=8) (DOUBLE PRECISION) Representation COMPLEX(KIND=4) (COMPLEX) Representation COMPLEX(KIND=8) (DOUBLE COMPLEX) Representation
On using the BitViewer tool, see Viewing Floating-Point Representations with BitViewer. Reading or writing floating-point data other than native IEEE little endian data, see Converting Unformatted Numeric Data. Using floating-point numbers, see The Floating-Point Environment.
Data Representation
Page 7 of 9
The form of REAL(4) data is sign magnitude, with bit 31 the sign bit (0 for positive numbers, 1 for negative numbers), bits 30:23 a binary exponent in excess 127 notation, and bits 22:0 a normalized 24-bit fraction including the redundant most-significant fraction bit not represented. The value of data is in the approximate range: 1.17549435E-38 (normalized) to 3.40282347E38. The IEEE denormalized (subnormal) limit is 1.40129846E-45. The precision is approximately one part in 2**23; typically 7 decimal digits.
The form of REAL(8) data is sign magnitude, with bit 63 the sign bit (0 for positive numbers, 1 for negative numbers), bits 62:52 a binary exponent in excess 1023 notation, and bits 51:0 a normalized 53-bit fraction including the redundant most-significant fraction bit not represented. The value of data is in the approximate range: 2.2250738585072013D-308 (normalized) to 1.7976931348623158D308. The IEEE denormalized (subnormal) limit is 4.94065645841246544D-324. The precision is approximately one part in 2**52; typically 15 decimal digits.
Data Representation
Page 8 of 9
The limits and underflow characteristics for REAL(4) apply to the two separate real and imaginary parts of a COMPLEX(4) number. Like REAL(4) numbers, the sign bit representation is 0 (zero) for positive numbers and 1 for negative numbers.
The limits and underflow characteristics for REAL(8) apply to the two separate real and imaginary parts of a COMPLEX(8) number. Like REAL(8) numbers, the sign bit representation is 0 (zero) for positive numbers and 1 for negative numbers.
Character Representation
A character string is a contiguous sequence of bytes in memory, as shown below. CHARACTER Data Representation
Data Representation
Page 9 of 9
A character string is specified by two attributes: the address A of the first byte of the string, and the length L of the string in bytes. The length L of a string is in the range 1 through 2,147,483,647 (2**31-1) .
Hollerith Representation
Hollerith constants are stored internally, one character per byte, as shown below. Hollerith Data Representation
Page 1 of 34
Default Run-Time Error Processing, which describes the message format and values returned at program termination. Methods of Handling Errors, which describes using the END, EOR, and ERR I/O statement branch specifiers and the IOSTAT specifier. Locating Run-Time Errors, which describes suggested compiler options and information about debugging exceptions. Using Traceback Information, which describes techniques to help you locate the cause of severe run-time errors. Obtaining Traceback Information with TRACEBACKQQ, which explains how to call a routine to obtain traceback information from your application. Run-Time Environment Variables, which describes environment variables that allow program continuation under certain conditions, disable the display of certain dialog boxes under certain conditions, and allow just-intime debugging.
During execution, your program may encounter errors or exception conditions. These conditions can result from any of the following:
l l l l l
Errors that occur during I/O operations Invalid input data Argument errors in calls to the mathematical library Arithmetic errors Other system-detected errors
The Visual Fortran run-time system (Run-Time Library or RTL) generates appropriate messages and takes action to recover from errors whenever possible. For a description of each Visual Fortran run-time error message, see Run-Time Errors.
Page 2 of 34
The severity of the error. For instance, the program usually continues executing when an error message with a severity level of warning or info (informational) is detected. For certain errors associated with I/O statements, whether or not an I/O error-handling specifier was specified. For certain errors, whether or not the default action of an associated signal was changed. For certain errors related to arithmetic operations (including floating-point exceptions), compilation options can determine whether the error is reported and the severity of the reported error.
How arithmetic exception conditions are reported and handled depends on the cause of the exception and how the program was compiled. Unless the program was compiled to handle exceptions, the exception might not be reported until after the instruction that caused the exception condition. For more information:
l
l l
About where Visual Fortran run-time message are displayed and their format, see Run-Time Message Display and Format. On the Visual Fortran return values at program termination, see Values Returned at Program Termination. On locating errors and the compiler options related to handling errors and exceptions, see Locating Run-Time Errors and Using Traceback Information. On DF command options and their categories in the visual development environment (Project Settings, Fortran tab), see Categories of Compiler Options. On Compaq Fortran intrinsic data types and their ranges, see Data Representation. On the floating-point environment, see The Floating-Point Environment. On creating exception and termination handlers, see Advanced Exception and Termination Handling Considerations.
Page 3 of 34
Where Fortran Run-Time Messages Appear Run-time error messages are displayed on the standard error device (unless redirected). Run-time error messages are displayed in a separate QuickWin message box. Run-time error messages are displayed in a separate message box.
Fortran QuickWin and Fortran Standard Graphics applications Fortran Windows applications
Fortran run-time messages have the following format: forrtl: severity (number): message-text where:
l
forrtl Identifies the source as the Visual Fortran run-time system (Run-Time Library or RTL). severity The severity levels are: severe, error, warning, or info (abbreviation of information). number This is the message number, also the IOSTAT value for I/O statements. message-text Explains the event that caused the message.
The following table explains the severity levels of run-time messages, in the order of greatest to least severity. Severity Levels of Run-Time Messages Severity Description
Page 4 of 34
Must be corrected. The program's execution is terminated when the error is encountered unless the program's I/O statements use the END, EOR, or ERR branch specifiers to transfer control, perhaps to a routine that uses the IOSTAT specifier (see the section Using the END, EOR, and ERR Branch Specifiers and the section Using the IOSTAT Specifier and Fortran Exit Codes). For severe errors, a hexadecimal dump of the call stack (program counter trace) is displayed after the error message. If the /traceback option was specified, the displayed call stack program counter trace includes the source file name, routine name, and source line number of your Fortran source code.
Should be corrected. The program might continue execution, but the output from this execution may be incorrect. Should be investigated. The program continues execution, but output from this execution may be incorrect. For informational purposes only; the program continues.
For a description of each Visual Fortran run-time error message, see Visual Fortran Run-Time Errors
The program runs to normal completion. A value of zero is returned to the shell. The program stops with a STOP statement. A value of zero is returned to the shell. The program stops because of a signal that is caught but does not allow the program to continue. A value of 1 is returned to the shell. The program stops because of a severe run-time error. The error number for that run-time error is returned to the shell. Error numbers are listed in Visual Fortran Run-Time Errors. The program stops with a CALL EXIT statement. The value passed to EXIT is returned to the shell.
Page 5 of 34
To transfer control to error-handling code within the program, use the END, EOR, and ERR branch specifiers in I/O statements, see Using the END, EOR, and ERR Branch Specifiers. To identify Fortran-specific I/O errors based on the value of Visual Fortran RTL error codes, use the I/O status specifier (IOSTAT) in I/O statements (or call the ERRSNS subroutine), see Using the IOSTAT Specifier and Fortran Exit Codes.
These error-processing methods are complementary; you can use any or all of them within the same program to obtain Visual Fortran run-time error codes.
The END branch specifier handles an end-of-file condition. The EOR branch specifier handles an end-of-record condition for nonadvancing reads. The ERR branch specifier handles all error conditions.
If you use the END, EOR, or ERR branch specifiers, no error message is displayed and execution continues at the designated statement, usually an error-handling routine. You might encounter an unexpected error that the error-handling routine cannot handle. In this case, do one of the following:
l l
Modify the error-handling routine to display the error message number. Remove the END, EOR, or ERR branch specifiers from the I/O statement that causes the error.
After you modify the source code, compile, link, and run the program to display the error message. For example:
READ (8,50,ERR=400)
If any severe error occurs during execution of this statement, the Visual Fortran RTL transfers control to the statement at label 400. Similarly, you can use the END specifier to handle an end-of-file condition that might otherwise be treated as an error. For example:
READ (12,70,END=550)
Page 6 of 34
When using nonadvancing I/O, use the EOR specifier to handle the end-ofrecord condition. For example:
150 FORMAT (F10.2, F10.2, I6) READ (UNIT=20, FMT=150, SIZE=X, ADVANCE='NO', EOR=700) A, F, I
You can also use ERR as a specifier in an OPEN, CLOSE, or INQUIRE statement. For example:
OPEN (UNIT=10, FILE='FILNAM', STATUS='OLD', ERR=999)
If an error is detected during execution of this OPEN statement, control transfers to the statement at label 999.
A value of -2 if an end-of-record condition occurs with nonadvancing reads. A value of -1 if an end-of-file condition occurs. A value of 0 for normal completion (not an error condition, end-of-file, or end-of-record condition). A positive integer value if an error condition occurs. (This value is one of the Fortran-specific IOSTAT numbers listed in Visual Fortran Run-Time Errors.)
Following the execution of the I/O statement and assignment of an IOSTAT value, control transfers to the END, EOR, or ERR statement label, if any. If there is no control transfer, normal execution continues. For more information on transfer of control, see Branch Statements. You can include the iosdef.for file in your program to obtain symbolic definitions
Page 7 of 34
The following example uses the IOSTAT specifier and the iosdef.for file to handle an OPEN statement error (in the FILE specifier). Error Handling OPEN Statement File Name
CHARACTER(LEN=40) :: FILNM INCLUDE 'iosdef.for' DO I=1,4 FILNM = '' WRITE (6,*) 'Type file name ' READ (5,*) FILNM OPEN (UNIT=1, FILE=FILNM, STATUS='OLD', IOSTAT=IERR, ERR=100) WRITE (6,*) 'Opening file: ', FILNM (process the input file) CLOSE (UNIT=1) STOP 100 IF (IERR .EQ. FOR$IOS_FILNOTFOU) THEN WRITE (6,*) 'File: ', FILNM, ' does not exist ' ELSE IF (IERR .EQ. FOR$IOS_FILNAMSPE) THEN WRITE (6,*) 'File: ', FILNM, ' was bad, enter new file name' ELSE PRINT *, 'Unrecoverable error, code =', IERR STOP END IF END DO WRITE (6,*) 'File not found. Locate correct file with Explorer and run again' END PROGRAM
Within the DO loop, instead of the READ statement that assigns a value to variable FILNM, you could instead call the GetOpenFileName Win32 routine, which requests the file name using a dialog box. For an example of how to call the GetOpenFileName routine, see the Visual Fortran Sample GetOpenFileName in ...Df98\Samples\Advanced\Win32\GetOpenFileName.
The /[no]warn options control compile-time diagnostic messages, which in some circumstances can help determine the cause of a run-time error. In the Microsoft visual development environment, specify the Warning Level in the General Compiler Option Category or specify individual Warning Options in the Compiler Diagnostics Compiler Option Category.
Page 8 of 34
The /check:keyword options generate extra code to catch certain conditions at run-time (see /check or in the visual development environment, specify the Runtime Error Checking items in the Run time Compiler Option Category). For example:
The /check:bounds option generates extra code to catch access to data beyond the array or string boundaries. The /check:overflow option generates extra code to catch integer overflow conditions. The /check:noformat, /check:nooutput_conversion, and /check:nopower options reduce the severity level of the associated run-time error to allow program continuation (see /check). The /check:underflow option controls the reporting of floating-point underflow exceptions at run-time.
The /traceback option allows program counter to source file line correlation, which simplifies the task of locating the cause of severe runtime errors. Without /traceback, a link map file and source listing are usually needed to locate the cause of the error. Certain traceback-related information accompanies severe run-time errors, as described in Using Traceback Information. The /fpe option controls the handling of floating-point arithmetic exceptions (IEEE arithmetic) at run-time. In the visual development environment, specify the Floating-Point Exception Handling in the Floating Point Compiler Option Category. For example, if you specified /fpe:3, exceptions related to exceptional IEEE values are not reported and your application may generate exceptional IEEE values, which later in your application may generate an exception or unexpected values. By recompiling the application at /fpe:0, any exceptional IEEE values generated will cause the program to terminate and report an error message earlier. The FOR_GET_FPE and FOR_SET_FPE routines are also available to examine and set the run-time handling of certain arithmetic exceptions.
You can use the debugger to help you locate exceptions, as described in Locating Run-Time Errors in the Debugger.
Page 9 of 34
The Fortran run-time system attempts to walk back up the call chain and produce a report of the calling sequence leading to the error as part of the default diagnostic message report. The minimum information displayed includes:
l
The standard Fortran run-time error message text that explains the error condition. A tabular report that contains one line per call stack frame. This information includes at least the image name and a hexadecimal PC in that image.
The information displayed under the Routine, Line, and Source columns depends upon whether your program was compiled with the /traceback option. In the visual development environment, the default is /traceback in the debug configuration, but must be requested for the release configuration (Run Time category of Project settings). For example, if /traceback were specified, the displayed information might resemble the following:
forrtl: severe (24): end-of-file during read, unit 10, file E:\USERS\xxx.dat Image PC Routine Line Source DFORRT.dll 1000A3B2 Unknown Unknown Unknown DFORRT.dll 1000A184 Unknown Unknown Unknown DFORRT.dll 10009324 Unknown Unknown Unknown DFORRT.dll 10009596 Unknown Unknown Unknown DFORRT.dll 10024193 Unknown Unknown Unknown teof.exe 004011A9 AGAIN 21 teof.for teof.exe 004010DD GO 15 teof.for teof.exe 004010A7 WE 11 teof.for teof.exe 00401071 HERE 7 teof.for teof.exe 00401035 TEOF 3 teof.for teof.exe 004013D9 Unknown Unknown Unknown teof.exe 004012DF Unknown Unknown Unknown KERNEL32.dll 77F1B304 Unknown Unknown Unknown
If the same program was not compiled with the /traceback option (/notraceback):
l
The Routine name, Line number, and Source file columns would be reported as "Unknown." A link map file and source listing are usually needed to locate the cause of the error.
The /traceback option provides program counter to source file line correlation information to appear in the displayed error message information, which simplifies the task of locating the cause of severe run-time errors. For Fortran objects generated with /traceback, the compiler generates additional information used by the Fortran run-time system to automatically correlate PC (program counter) values to the routine name in which they occur, Fortran source file, and line number in the source file. This information is displayed in
Page 10 of 34
Automatic PC correlation is only supported for Fortran code. For non-Fortran code, only the hexadecimal PC locations are reported. The following sections describe traceback-related tools, lists traceback-related environment variables, discuss tradeoffs and restrictions, and provide examples:
l l l l
Tools to Help You Understand Traceback Output Relevant Fortran Run-Time Environment Variables Tradeoffs and Restrictions Example Programs and Traceback Information
When using the Microsoft visual development environment: 1. Choose Settings from the Project menu 2. Click the Fortran tab 3. Select the Listing Files category 4. Click (check) "Source Listing" 5. Click "Machine Code" 6. Click "Include Files" 7. Make any other changes needed to the Project Settings, then click OK. When compiling from the command line, specify /list with /show:all:
DF file.f90 /list /show:all
When using the visual development environment: 1. Choose Settings from the Project menu 2. Click the Link tab 3. Select the General or Debug category 4. Select Generate mapfile 5. Make any other changes needed to the Project Settings, then click OK. When compiling from the command line, specify /map:
Page 11 of 34
DF file.f90 /map
Specify the /traceback compile option. This requests that the compiler generate PC correlation information to be used at run-time in the event of an error. This information is gathered up by the linker in the executable image in a section named ".trace". To see what sections are in an image, use the command:
link -dump -summary your_app_name.exe
FOR_DIAGNOSTIC_LOG_FILE If set to the name of a file, writes diagnostic output to the specified file. The Fortran run-time system attempts to open that file (append output) and write the error information (ASCII text) to the file. The setting of FOR_DIAGNOSTIC_LOG_FILE is independent of FOR_DISABLE_DIAGNOSTIC_DISPLAY, so you can disable the screen display of information but still capture the error information in a file. The text string you assign for the file name is used literally so you must specify the full name. If the file open fails, no error is reported and the run-time system continues diagnostic processing.
FOR_DISABLE_DIAGNOSTIC_DISPLAY If set to true, disables the display of all error information. This might be helpful if you just want to test the error status of your program and do not want the Fortran run-time system to display any information about an abnormal program termination.
FOR_DISABLE_STACK_TRACE If set to true, disables the display of call stack information that follows the displayed severe error message text. The Fortran run-time error message
Page 12 of 34
FOR_ENABLE_VERBOSE_STACK_TRACE If set to true, more detailed call stack information is displayed in the event of an error. The default brief output is usually sufficient to determine where an error occurred. Brief output includes up to twenty stack frames, reported one line per stack frame. For each frame, the image name containing the PC, the PC, routine name, line number, and source file are given. The verbose output, if selected, will provide in addition to the information in brief output, the exception context record if the error was a machine exception (machine register dump), and for each frame, the return address, frame pointer and stack pointer and possible parameters to the routine. This output can be quite long (but limited to 16K bytes) and use of the environment variable FOR_DIAGNOSTIC_LOG_FILE is recommended if you want to capture the output accurately. Most situations should not require the use of verbose output.
FOR_FULL_SRC_FILE_SPEC By default, the traceback output displays only the file name and extension in the source file field. To display complete file name information including the path, set the environment variable FOR_FULL_SRC_FILE_SPEC to true.
Effect on image size using /traceback Using the /traceback option to get automatic PC correlation does increase the size of an image. For any application, the developer must decide if the increase in image size is worth the benefit of automatic PC correlation or if manually correlating PC's with a map file is acceptable. The approach of providing automatic correlation information in the image was used so that no run-time penalty is incurred by building the information "on the fly" as your application executes. No run-time diagnostic code is invoked unless your application is terminating due to a severe error.
The C Compiler Frame Pointer Omission Option (/Oy) on ia32 Systems At the heart of the stack walking code is the Win32 routine StackWalk( )
Page 13 of 34
found in imagehlp.dll. In the ia32 systems environment, there are no firm software calling standards documented. Compiler developers are under no constraints to use machine registers in any particular way or to hook up procedures in any particular way. The StackWalk( ) routine therefore, bases its decisions on how to walk the call stack on a set of heuristics. That is, it makes a "best guess" to determine how a program reached a particular point in the call chain. With C code that has been compiled with Visual C++ with the Frame Pointer Omission Option (/Oy) enabled, this "best guess" is not usually the correct one. If you are mixing Fortran and C code and you are concerned about stack tracing, consider disabling this option (/Oy-) in your C compiles. Otherwise traceback will most likely not work for you.
l
Linker /incremental:no option When incremental linking is enabled, automatic PC correlation does not work. Use of incremental linking always disables automatic PC correlation even if you specify /traceback during compilation. When you use incremental linking, the default hexadecimal (hex) PC values will still appear in the output. To correlate from the hexadecimal PC values to routine containing the PC addresses requires use of a linker map file. However, if you request a map file during linking, incremental linking becomes disabled. Thus to allow any PC values generated for a run-time problem to be helpful, incremental linking must be disabled. In the visual development environment, you can use the Call stack display, so incremental linking is not a problem.
When the Stack Trace Fails For Some Reason Programs can fail for a multitude of reasons with unpredictable consequences. Memory corruption by erroneously executing code is always a possibility. Stack memory may be corrupted in such a way that the attempt to trace the call stack will result in access violations or other undesirable consequences. The stack tracing run-time code is guarded with a local exception filter. Should the traceback attempt fail due to a hard detectable condition, the run-time will report this in its diagnostic output message as:
Stack trace terminated abnormally
Be forewarned, however, it is also possible for memory to be corrupted in such a way that a stack trace can seem to complete successfully with no hint of a problem. The bit patterns it finds in corrupted memory where the stack used to be, and then uses to access memory, may constitute
Page 14 of 34
perfectly valid memory addresses for the program to be accessing. They just do not happen to have any connection to what the stack used to look like. So, if it appears that the stack walk completed normally, but the reported PC's make no sense to you, then consider ignoring the stack trace output in diagnosing your problem. You may also see the stack trace fail if the run-time system cannot dynamically load imagehlp.dll or cannot find the routines from that library it needs to do the stack walk. In this case, you would still get the basic runtime diagnostic message. You just will not get any call stack information. Another condition that will disable the stack trace process is your program exiting because it has exhausted virtual memory resources.
Example: End-of-File Condition, Program teof Example: Machine Exception Condition, Program ovf Example: Using Traceback in Mixed Fortran/C Applications, Program FPING and CPONG
Example: End-of-File Condition, Program teof In the following example, a READ statement creates an End-Of-File error, which the application has not handled:
program teof integer*4 i,res i=here( ) end integer*4 function here( ) here = we( ) end integer*4 function we( ) we = go( ) end integer*4 function go( ) go = again( ) end integer*4 function again( ) integer*4 a open(10,file='xxx.dat',form='unformatted',status='unknown') read(10) a again=a end
Page 15 of 34
The diagnostic output that results when this program is built with traceback enabled and linked against the single-threaded, DLL Fortran run-time library on the ia32 platform:
forrtl: severe (24): end-of-file during read, unit 10, file E:\USERS\xxx.dat Image PC Routine Line Source DFORRT.dll 1000A3B2 Unknown Unknown Unknown DFORRT.dll 1000A184 Unknown Unknown Unknown DFORRT.dll 10009324 Unknown Unknown Unknown DFORRT.dll 10009596 Unknown Unknown Unknown DFORRT.dll 10024193 Unknown Unknown Unknown teof.exe 004011A9 AGAIN 21 teof.for teof.exe 004010DD GO 15 teof.for teof.exe 004010A7 WE 11 teof.for teof.exe 00401071 HERE 7 teof.for teof.exe 00401035 TEOF 3 teof.for teof.exe 004013D9 Unknown Unknown Unknown teof.exe 004012DF Unknown Unknown Unknown KERNEL32.dll 77F1B304 Unknown Unknown Unknown
The first line of the output is the standard Fortran run-time error message. What follows is the result of walking the call stack in reverse order to determine where the error originated. Each line of output represents a call frame on the stack. Since the application was compiled with /traceback, the PC's that fall in Fortran code are correlated to their matching routine name, line number and source module. PC's which are not in Fortran code are not correlated and are reported as "Unknown." The first five frames show the calls to routines in the Fortran run-time library (in reverse order). Since the application was linked against the single threaded DLL version of the library, the image name reported is DFORRT.dll. These are the run-time routines that were called to do the READ and upon detection of the EOF condition, were invoked to report the error. In the case of an unhandled I/O programming error, there will always be a few frames on the call stack down in run-time code like this. The stack frame of real interest to the Fortran developer is the first frame in image teof.exe which shows that the error originated in the routine named AGAIN in source module teof.for at line 21. Looking in the source code at line 21, we see the Fortran READ statement that incurred the end-of-file condition. The next four frames show that completes the trail of calls in the Fortran user code that led to the routine that got the error (TEOF->HERE->WE->GO>AGAIN). Finally, the bottom three frames are routines which handled the startup and initialization of the program. If this program had been linked against the single-threaded, static Fortran runtime library, the output would then look like:
Page 16 of 34
forrtl: severe Image teof.exe teof.exe teof.exe teof.exe teof.exe teof.exe teof.exe teof.exe teof.exe teof.exe teof.exe teof.exe KERNEL32.dll
(24): end-of-file during read, unit 10, file E:\USERS\xxx.dat PC Routine Line Source 004067D2 Unknown Unknown Unknown 0040659F Unknown Unknown Unknown 00405754 Unknown Unknown Unknown 004059C5 Unknown Unknown Unknown 00403543 Unknown Unknown Unknown 004011A9 AGAIN 21 teof.for 004010DD GO 15 teof.for 004010A7 WE 11 teof.for 00401071 HERE 7 teof.for 00401035 TEOF 3 teof.for 004202F9 Unknown Unknown Unknown 00416063 Unknown Unknown Unknown 77F1B304 Unknown Unknown Unknown
Notice that the initial five stack frames now show routines in image teof.exe, not DFORRT.DLL. The routines are the same five run-time routines as previously reported for the DLL case but since the application was linked against the static Fortran run-time library (dfor.lib), the object modules containing these routines were linked into the application image (teof.exe). Using the map file, you can determine that the PC reported in the top stack frame, 004067D2, is from routine _for_stack_trace, in module for_diags.obj of library dfor.lib (004067a0 < PC=004067D2 < 004079f0):
... 0001:000057a0 0001:000069f0 ... _for_stack_trace 004067a0 f dfor:for_diags.obj _for__find_trace_info_file 004079f0 f dfor:for_diags.obj
Now suppose the application was compiled without traceback enabled and once again, linked against the single-threaded, static Fortran library. The diagnostic output would then appear as follows:
forrtl: severe (24): end-of-file during read, unit 10, file E:\USERS\xxx.dat Image PC Routine Line Source teof.exe 00406792 Unknown Unknown Unknown teof.exe 0040655F Unknown Unknown Unknown teof.exe 00405714 Unknown Unknown Unknown teof.exe 00405985 Unknown Unknown Unknown teof.exe 00403503 Unknown Unknown Unknown teof.exe 00401169 Unknown Unknown Unknown teof.exe 004010A8 Unknown Unknown Unknown teof.exe 00401078 Unknown Unknown Unknown teof.exe 00401048 Unknown Unknown Unknown teof.exe 0040102F Unknown Unknown Unknown teof.exe 004202B9 Unknown Unknown Unknown teof.exe 00416023 Unknown Unknown Unknown KERNEL32.dll 77F1B304 Unknown Unknown Unknown
Without the correlation information in the image that /traceback previously supplied, the Fortran run-time system cannot correlate PC's to routine name, line number, and source file. You can still use the map file to at least determine the routine names and what modules they are in. Look at the beginning of the entry point list in the map file for this image:
Page 17 of 34
Address 0001:00000000 0001:00000000 0001:00000000 0001:0000003d 0001:0000003d 0001:0000006d 0001:0000006d 0001:0000009d 0001:0000009d 0001:000000cd 0001:000000cd 0001:00000180 ...
Publics by Value _TEOF _TEOF@0 _MAIN__ _HERE _HERE@0 _WE@0 _WE _GO _GO@0 _AGAIN _AGAIN@0 _for_rtl_init_
Rva+Base 00401000 00401000 00401000 0040103d 0040103d 0040106d 0040106d 0040109d 0040109d 004010cd 004010cd 00401180 f f f f f f f f f f f f
Lib:Object teof.obj teof.obj teof.obj teof.obj teof.obj teof.obj teof.obj teof.obj teof.obj teof.obj teof.obj dfor:for_init.obj
After determining that the first five stack frames fall in run-time code (using portions of the map file not shown here), the sixth frame shows a PC of 00401169. Using the fragment of the map file shown above, it can be seen that this PC is greater than 004010cd but less than 00401180 and is therefore in routine _AGAIN, the first Fortran routine on the call stack. The remaining PC's can be manually correlated in a similar fashion to reconstruct the calling sequence. Remember that compiling with /traceback increases the size of your application's image because of the extra PC correlation information included in the image. You can see if the extra traceback information is included in an image (checking for the presence of a .trace section) by typing:
link -dump -summary your_app.exe
Check the file size with a simple directory command. Here's the teof.exe example linked against the dynamic Fortran library with traceback:
03/03/98 01:45p 1 File(s) 5,120 teof.exe 5,120 bytes
Page 18 of 34
For this simple example, the traceback correlation information added 512 bytes to the image size. In a real application, this would probably be much larger. For any application, the developer must decide if the increase in image size is worth the benefit of automatic PC correlation or if manually correlating PC's with a map file is acceptable. For command-line use and in the release configuration in the visual development environment, traceback information is not included by default in Fortran compiles (default is /notraceback). In the visual development environment, the default is /traceback in the debug configuration. For the release configuration, request traceback in the Run Time category of the Project Settings dialog box. If an error occurs when traceback was requested during compilation, the runtime library will produce the correlated call stack display. If an error occurs when traceback was disabled during compilation, the run-time library will produce the uncorrelated call stack display. If you do not want to see the call stack information displayed, you can set the environment variable FOR_DISABLE_STACK_TRACE to true. You will still get the Fortran run-time error message:
forrtl: severe (24): end-of-file during read, unit 10, file E:\USERS\xxx.dat
Example: Machine Exception Condition, Program ovf The following program generates a floating-point overflow exception when compiled with /fpe:0:
program ovf real*4 a a=1e37 do i=1,10 a=hey(a) end do print *, 'a= ', a end real*4 function hey(b) real*4 b hey = watch(b) end real*4 function watch(b) real*4 b
Page 19 of 34
When this program is compiled on an ia32 system with /traceback, /fpe:0, and /optimization:0, the traceback output appears as follows:
forrtl: error (72): floating Image PC ovf.exe 00401161 ovf.exe 0040113C ovf.exe 0040111B ovf.exe 004010FA ovf.exe 0040105B ovf.exe 00432429 ovf.exe 00426C74 KERNEL32.dll 77F1B9EA overflow Routine BELOW OUT WATCH HEY OVF Unknown Unknown Unknown Line 29 24 19 14 7 Unknown Unknown Unknown Source ovf.f90 ovf.f90 ovf.f90 ovf.f90 ovf.f90 Unknown Unknown Unknown
Notice that unlike the previous example of an unhandled I/O programming error, the stack walk can begin right at the point of the exception. There are no run-time routines on the call stack to dig through. The overflow occurs in routine BELOW at PC 00401161 which is correlated to line 29 of the source file ovf.f90. When the program is compiled at a higher optimization level, /optimization:4, with /fpe:0 and /traceback, the traceback output appears as follows:
forrtl: error (72): floating Image PC ovf.exe 00401070 ovf.exe 004323E9 ovf.exe 00426C34 KERNEL32.dll 77F1B9EA overflow Routine OVF Unknown Unknown Unknown
With /optimize:4, the entire program has been inlined. We can see this with a quick look in the listing file:
... 0000 0000 0001 0003 0006 000B 000E .CODE PUBLIC _MAIN__ _MAIN__ PROC push ebp mov ebp, esp sub esp, 48 call _for_check_flawed_pentium@0 sub lea esp, 4 eax, dword ptr .literal$+32
Page 20 of 34
0014 0015 001A 0020 0021 0026 0029 002F 0035 0036 003C 003D 0043 0044 004A 004B 0051 0052 0058 0059 005F 0060 0066 0067 006D 0070 0071 0073 0079 0080 0083 0089 008C 008F 0092 0095 0096 0097 009C 00A1 00A2 00A7 00AA 00AD
push call lea push call add fld fmul fwait fmul fwait fmul fwait fmul fwait fmul fwait fmul fwait fmul fwait fmul fwait fmul fstp fwait xor lea mov mov lea lea mov sub lea push push push push push call add mov mov
eax _for_set_fpe_@4 eax, dword ptr .literal$+28 eax _for_set_reentrancy@4 esp, 4 dword ptr .literal$+24 dword ptr .literal$ dword ptr .literal$ dword ptr .literal$ dword ptr .literal$ dword ptr .literal$ dword ptr .literal$ dword ptr .literal$ dword ptr .literal$ dword ptr .literal$ dword ptr -4[ebp] edx, edx eax, dword ptr .literal$+20 dword ptr -48[ebp], 3 dword ptr -44[ebp], eax ecx, dword ptr .literal$+12 eax, .T1_ ; eax, dword ptr -36[ebp] .T1_, edx ; dword ptr -36[ebp], edx esp, 4 edx, dword ptr -48[ebp] edx ecx 126156544 -1 eax _for_write_seq_lis esp, 24 eax, dword ptr -4[ebp] dword ptr -48[ebp], eax ; 000029
Page 21 of 34
The main program, OVF, no longer calls routine HEY. While the output is not quite what one might have expected intuitively, it is still entirely correct. You need to keep in mind the effects of compiler optimization when you interpret the diagnostic information reported for a failure in a release image. If the same image were executed again but with environment variable FOR_ENABLE_VERBOSE_STACK_TRACE set to true, you would also see a dump of the exception context record at the time of the error. Here is an excerpt of how that would appear on an ia32 system:
forrtl: error (72): floating overflow Hex Dump Of Exception Record Context Information: Exception Context: EFlags: 00010212 CS: 0000001B EIP: Exception Context: EAX: 00444488 ESI: 0012FBBC EBX: EDI: Processor Control and Status Registers. 00401161 SS: 00000023 ESP: 0012FE38 EBP: 0012FE60
Floating Point Control and Status Registers. ErrorOffset: ErrorSelector: Cr0NpxState: 0040115E 015D001B 00000000 DataOffset: DataSelector: 0012FE5C FFFF0023
Floating Point RegisterArea. 4080BC143F4000000000 00131EF0000800060012 02080006000000000000 00000000000000300000 RegisterArea[10]: RegisterArea[30]: RegisterArea[50]: RegisterArea[70]: F7A0FFFFFFFF77F9D860 00000012F7C002080006 0000000000000012F7D0 FBBC000000300137D9EF
Page 22 of 34
Example: Using Traceback in Mixed Fortran/C Applications, Program FPING and CPONG Consider the following example that shows how the traceback output might appear in a mixed Fortran/C application. The main program is a Fortran program named FPING. Program FPING triggers a chain of function calls which are alternately Fortran and C code. Eventually, the C routine named Unlucky is called, which produces a floating divide by zero error. Source module FPING.FOR contains the Fortran function definitions, each of which calls a C routine from source module CPONG.C. FPING.FOR is compiled with the /fpe:0 /optimize:0 /traceback options enabled, on the ia32 platform. Here's the program traceback output:
forrtl: error (73): floating divide by zero Image PC Routine Line fping.exe 00401161 Unknown Unknown fping.exe 004010DC DOWN4 58 fping.exe 0040118F Unknown Unknown fping.exe 004010B6 DOWN3 44 fping.exe 00401181 Unknown Unknown fping.exe 00401094 DOWN2 31 fping.exe 00401173 Unknown Unknown fping.exe 00401072 DOWN1 18 fping.exe 0040104B FPING 5 fping.exe 004013B9 Unknown Unknown fping.exe 004012AF Unknown Unknown KERNEL32.dll 77F1B304 Unknown Unknown Source Unknown fping.for Unknown fping.for Unknown fping.for Unknown fping.for fping.for Unknown Unknown Unknown
Notice that the stack frames contributed by Fortran routines can be correlated to a routine name, line number, and source module but those frames contributed by C routines cannot be correlated. Remember, even though the stack can be walked in reverse, and PC's reported, the information necessary to correlate the PC to a routine name, line number, and so on, is contributed to the image from the objects generated by the Fortran compiler. The C compiler does not have this capability. Also remember that you only get the correlation information if you specify the /traceback option for your Fortran compiles. The top stack frame cannot be correlated to a routine name because it is in C code. By examining the map file for the application, we can see that the reported PC, 00401161, is greater than the start of routine _Unlucky, but less than the start of routine _down1_C, so we at least know the error occurred in routine _Unlucky. Here is the pertinent section of the map file:
Address 0001:00000000 0001:00000000 0001:00000000 0001:0000005f 0001:0000005f Publics by Value _FPING _FPING@0 _MAIN__ _DOWN1 _DOWN1@4 Rva+Base 00401000 00401000 00401000 0040105f 0040105f f f f f f Lib:Object fping.obj fping.obj fping.obj fping.obj fping.obj
Page 23 of 34
****************************************************** The reported PC lies between the start of _Unlucky and the start of _down1_C... ****************************************************** 0001:0000014d 0001:00000167 0001:00000175 0001:00000183 0001:00000192 0001:00000198 0001:0000019e etc... _Unlucky _down1_C 0040114d f cpong.obj 00401167 f cpong.obj
_down2_C 00401175 f cpong.obj _down3_C 00401183 f cpong.obj _for_check_flawed_pentium@0 00401192 f dfordll:DFORRT.dll _for_set_fpe_@4 00401198 f dfordll:DFORRT.dll _for_set_reentrancy@4 0040119e f dfordll:DFORRT.dll
In a similar manner, the other PC's reported as "Unknown" can be correlated to a routine name using the map file. When examining traceback output (or any type of diagnostic output for that matter), it is always important to keep in mind the affects of compiler optimization. The Fortran source module in the above example was built with optimization turned off, /optimize:0. Look at the output when optimizations are enabled, /optimize:4:
forrtl: error (73): floating divide by zero Image PC Routine Line fping.exe 00401111 Unknown Unknown fping.exe 0040109D DOWN4 58 fping.exe 0040113F Unknown Unknown fping.exe 00401082 DOWN3 44 fping.exe 00401131 Unknown Unknown fping.exe 0040106B DOWN2 31 fping.exe 00401123 Unknown Unknown fping.exe 00401032 FPING 18 fping.exe 00401369 Unknown Unknown fping.exe 0040125F Unknown Unknown KERNEL32.dll 77F1B304 Unknown Unknown Source Unknown fping.for Unknown fping.for Unknown fping.for Unknown fping.for Unknown Unknown Unknown
From the traceback output, it would appear that routine DOWN1 was never called. In fact, it has not been called. At the higher optimization level, the compiler has inlined function DOWN1 so that the call to routine down1_C is now made from FPING. The correlated line number still points to the correct line in the source code. You can see that DOWN1 was inlined by looking in the listing file, FPING.LST:
.CODE PUBLIC _MAIN__ PROC push _MAIN__ ebp
55
0000 0000
Page 24 of 34
; 000018 ; 000018
******************************************************************* Call _down1_C from MAIN__ here, no call to DOWN1 Fortran routine... ******************************************************************* 000000E8 00 C0DD F7D9 08C483 000001B8 A0 5B E58B 5D C3 _MAIN__ 002D 0032 0034 0036 0039 003E 003F 0041 0042 call _down1_C
ffree st(0) fincstp add esp, 8 mov eax, 1 pop mov pop ret ENDP ebx esp, ebp ebp
; 000006
Finally, suppose the example Fortran code is redesigned with each of the Fortran routines split into separate source modules. Here is what the traceback output would look like with the redesigned code:
forrtl: error (73): floating divide by zero Image PC Routine Line fpingmain.exe 00401171 Unknown Unknown fpingmain.exe 004010ED DOWN4 12 fpingmain.exe 0040119F Unknown Unknown fpingmain.exe 004010C1 DOWN3 11 fpingmain.exe 00401191 Unknown Unknown fpingmain.exe 00401099 DOWN2 11 fpingmain.exe 00401183 Unknown Unknown fpingmain.exe 00401073 DOWN1 11 fpingmain.exe 0040104B FPING 5 fpingmain.exe 004013C9 Unknown Unknown fpingmain.exe 004012BF Unknown Unknown KERNEL32.dll 77F1B304 Unknown Unknown Source Unknown fping4.for Unknown fping3.for Unknown fping2.for Unknown fping1.for fpingmain.for Unknown Unknown Unknown
Notice that the line number and source file correlation information has changed to reflect the new design of the code. Here are the sources used in the above examples:
Page 25 of 34
************************************ FPING.FOR ************************************ program fping real*4 a,b a=-10.0 b=down1(a) end real*4 function down1(b) real*4 b !DEC$ IF DEFINED(_X86_) INTERFACE TO REAL*4 FUNCTION down1_C [C,ALIAS:'_down1_C'] (n) !DEC$ ELSE INTERFACE TO REAL*4 FUNCTION down1_C [C,ALIAS:'down1_C'] (n) !DEC$ ENDIF REAL*4 n [VALUE] END real*4 down1_C down1 = down1_C(b) end real*4 function down2(b) real*4 b [VALUE] !DEC$ IF DEFINED(_X86_) INTERFACE TO REAL*4 FUNCTION down2_C [C,ALIAS:'_down2_C'] (n) !DEC$ ELSE INTERFACE TO REAL*4 FUNCTION down2_C [C,ALIAS:'down2_C'] (n) !DEC$ ENDIF REAL*4 n [VALUE] END real*4 down2_C down2 = down2_C(b) end real*4 function down3(b) real*4 b [VALUE] !DEC$ IF DEFINED(_X86_) INTERFACE TO REAL*4 FUNCTION down3_C [C,ALIAS:'_down3_C'] (n) !DEC$ ELSE INTERFACE TO REAL*4 FUNCTION down3_C [C,ALIAS:'down3_C'] (n) !DEC$ ENDIF REAL*4 n [VALUE] END real*4 down3_C down3 = down3_C(b) end real*4 function down4(b) real*4 b [VALUE] !DEC$ IF DEFINED(_X86_) INTERFACE TO SUBROUTINE Unlucky [C,ALIAS:'_Unlucky'] (a,c) !DEC$ ELSE INTERFACE TO SUBROUTINE Unlucky [C,ALIAS:'Unlucky'] (a,c) !DEC$ ENDIF REAL*4 a [VALUE] REAL*4 c [REFERENCE] END real*4 a call Unlucky(b,a) down4 = a end
Page 26 of 34
************************************ CPONG.C ************************************ #include <math.h> extern float __stdcall DOWN2 (float n); extern float __stdcall DOWN3 (float n); extern float __stdcall DOWN4 (float n); int Fact( int n ) { if (n > 1) return( n * Fact( n - 1 )); return 1; } void Pythagoras( float a, float b, float *c) { *c = sqrt( a * a + b * b ); } void Unlucky( float a, float *c) { float b=0.0; *c = a/b; } float down1_C( float a ) { return( DOWN2( a )); } float down2_C( float a ) { return( DOWN3( a )); } float down3_C( float a ) { return( DOWN4( a )); } ************************************ FPINGMAIN.FOR ************************************ program fping real*4 a,b a=-10.0 b=down1(a) end ************************************ FPING1.FOR ************************************ real*4 function down1(b) real*4 b !DEC$ IF DEFINED(_X86_) INTERFACE TO REAL*4 FUNCTION down1_C [C,ALIAS:'_down1_C'] (n) !DEC$ ELSE INTERFACE TO REAL*4 FUNCTION down1_C [C,ALIAS:'down1_C'] (n) !DEC$ ENDIF
Page 27 of 34
************************************ FPING2.FOR ************************************ real*4 function down2(b) real*4 b [VALUE] !DEC$ IF DEFINED(_X86_) INTERFACE TO REAL*4 FUNCTION down2_C [C,ALIAS:'_down2_C'] (n) !DEC$ ELSE INTERFACE TO REAL*4 FUNCTION down2_C [C,ALIAS:'down2_C'] (n) !DEC$ ENDIF REAL*4 n [VALUE] END real*4 down2_C down2 = down2_C(b) end ************************************ FPING3.FOR ************************************ real*4 function down3(b) real*4 b [VALUE] !DEC$ IF DEFINED(_X86_) INTERFACE TO REAL*4 FUNCTION down3_C [C,ALIAS:'_down3_C'] (n) !DEC$ ELSE INTERFACE TO REAL*4 FUNCTION down3_C [C,ALIAS:'down3_C'] (n) !DEC$ ENDIF REAL*4 n [VALUE] END real*4 down3_C down3 = down3_C(b) end ************************************ FPING4.FOR ************************************ real*4 function down4(b) real*4 b [VALUE] !DEC$ IF DEFINED(_X86_) INTERFACE TO SUBROUTINE Unlucky [C,ALIAS:'_Unlucky'] (a,c) !DEC$ ELSE INTERFACE TO SUBROUTINE Unlucky [C,ALIAS:'Unlucky'] (a,c) !DEC$ ENDIF REAL*4 a [VALUE] REAL*4 c [REFERENCE] END real*4 a call Unlucky(b,a) down4 = a end
Page 28 of 34
TRACEBACKQQ allows an application to initiate a stack trace. You can use this routine to report application detected errors, use it for debugging, and so on. It uses the standard stack trace support in the Visual Fortran run-time system to produce the same output that the run-time system produces for unhandled errors and exceptions (severe error message). The TRACEBACKQQ subroutine generates a stack trace showing the program call stack as it was leading up to the point of the call to TRACEBACKQQ. The error message string normally included from the run-time support is replaced with the user-supplied message text or omitted if no user string is specified. Traceback output is directed to the target destination appropriate for the application type, just as it is when traceback is initiated internally by the run-time support. See the sections Using Traceback Information and Run-Time Message Display and Format for more information. In the most simple case, a user can generate a stack trace by coding the call to TRACEBACKQQ with no arguments:
CALL TRACEBACKQQ()
This call causes the run-time library to generate a traceback report with no leading header message, from wherever the call site is, and terminate execution. You can specify arguments that generate a stack trace with the user-supplied string as the header and instead of terminating execution, return control to the caller to continue execution of the application. For example:
CALL TRACEBACKQQ(STRING="Done with pass 1",USER_EXIT_CODE=-1)
By specifying a user exit code of -1, control returns to the calling program. Specifying a user exit code with a positive value requests that specified value be returned to the operating system. The default value is 0, which causes the application to abort execution. The following example demonstrates the use of the EPTR argument when calling from a user defined exception filter function. The premise of the example is a Fortran DLL containing entry points protected by a C try/except construct, such as:
__declspec(dllexport) void FPE_TEST_WRAPPER () { __try { /* ** call the Fortran code... */ FPE_TEST() ; }
Page 29 of 34
The C function shown above is guarding against a floating-point divide-by-zero exception. This function calls the user-supplied FPE_TEST (not shown). The Fortran function CHECK_EXCEPTION_INFO shown below (called in the __except filter expression above) can use TRACEBACKQQ to get a stack trace as follows:
INTEGER*4 FUNCTION CHECK_EXCEPTION_INFO ( ExceptionInfo ) !DEC$ ATTRIBUTES DLLEXPORT::CHECK_EXCEPTION_INFO USE DFWINTY !DEC$ ATTRIBUTES REFERENCE :: ExceptionInfo TYPE(T_EXCEPTION_POINTERS) ExceptionInfo TYPE(T_EXCEPTION_RECORD) erecptr TYPE(T_CONTEXT) ctxptr POINTER(eptr,erecptr) POINTER(ctxp,ctxptr) INTEGER(4) EXIT_CODE,STS CHARACTER(LEN=70) MYSTR ! Init the arguments to TRACEBACKQQ appropriately for your needs... EXIT_CODE=-1 eptr = ExceptionInfo.ExceptionRecord ctxp = ExceptionInfo.ContextRecord IF ( erecptr.ExceptionCode .EQ. STATUS_FLOAT_DIVIDE_BY_ZERO ) THEN PRINT *, 'Saw floating divide by zero.' PRINT '(1x,a,z8.8)', 'ContextRecord.FloatSave.StatusWord = ', & ctxptr.FloatSave.StatusWord MYSTR = "FLTDIV EXCEPTION IN MY APPLICATION" CALL TRACEBACKQQ(MYSTR,EXIT_CODE,STS, %LOC(ExceptionInfo)) END IF . . . CHECK_EXCEPTION_INFO = 1 END
To return a pointer to C run-time exception information pointers within a userdefined handler that was established with SIGNALQQ (or directly with the C signal function), you can call the GETEXCEPTIONPTRSQQ routine. For an example of how to establish a signal handler with SIGNALQQ (or the C signal() function) and use the TRACEBACKQQ and GETEXCEPTIONPTRSQQ routines to generate a traceback display from the handler, see the Visual Fortran Sample GetEptrs (in ...\Df98\Samples\ExceptionHandling\GetEptrs).
Page 30 of 34
Description The ACCEPT statement does not include an explicit logical unit number. Instead, it uses an implicit internal logical unit number and the FOR_ACCEPT environment variable. If FOR_ACCEPT is not defined, the code ACCEPT f,iolist reads from CONIN$ (standard input). If FOR_ACCEPT is defined (as a filename optionally containing a path), the specified file would be read. Lets you request that buffered I/O should be used at run-time for output of all Fortran I/O units, except those with output to the terminal. This provides a run-time mechanism to support the /assume:buffered_io compiler option. Lets you specify the print device other than the default print device PRN (LPT1) for files closed (CLOSE statement) with the DISPOSE='PRINT' specifier. To specify a different print device for the file associated with the CLOSE statement DISPOSE='PRINT' specifier, set the environment variable FOR_DEFAULT_PRINT_DEVICE to any legal DOS print device before executing the program. If set to the name of a file, writes diagnostic output to the specified file. For information on using stack trace information, see Locating Run-Time Errors and Using Traceback Information.
FORT_BUFFERED
FOR_DEFAULT_PRINT_DEVICE
FOR_DIAGNOSTIC_LOG_FILE
Page 31 of 34 If set to true, disables the display of all error information. This might be helpful if you just want to test the error status of your program and do not want the Fortran run-time system to display any information about an abnormal program termination. For information on using stack trace information, see Using Traceback Information. If set to true, disables the call stack trace information that follows the displayed severe error message text. For information on locating the cause of run-time errors using stack trace information, see Locating Run-Time Errors and Using Traceback Information. If set to true, displays more detailed call stack information in the event of an error. For information on using stack trace information, see Using Traceback Information. By default, the traceback output displays only the file name and extension in the source file field. To display complete file name information including the path, set the environment variable FOR_FULL_SRC_FILE_SPEC to true. For more information, see Using Traceback Information. In Visual Fortran Version 6, you no longer need to set this environment variable for the program to stop in the debugger when a severe error occurs. Regardless of whether this environment variable is set, you can view the Call Stack display. For more information, see Locating Run-Time Errors in the Debugger.
FOR_DISABLE_STACK_TRACE
FOR_ENABLE_VERBOSE_STACK_TRACE
FOR_FULL_SRC_FILE_SPEC
FOR_GENERATE_DEBUG_EXCEPTION
Page 32 of 34 If set to true, disables the default run-time exception handling, for example, to allow just-in-time debugging. The run-time system exception handler returns EXCEPTION_CONTINUE_SEARCH to the operating system, which looks for other handlers to service the exception. For information on just-in-time debugging, see Running Fortran Applications and the Visual C++ Development Environment User's Guide. If set to true, disables the display of dialog boxes when certain exceptions or errors occur. This is useful when running many test programs in batch mode to prevent a failure from stopping execution of the entire test stream. Neither the PRINT statement nor a WRITE statement with an asterisk (*) in place of a unit number includes an explicit logical unit number. Instead, both use an implicit internal logical unit number and the FOR_PRINT environment variable. If FOR_PRINT is not defined, the code PRINT f,iolist or WRITE (*,f) iolist writes to CONOUT$ (standard output). If FOR_PRINT is defined (as a filename optionally containing a path), the specified file would be written to. A READ statement that uses an asterisk (*) in place of a unit number does not include an explicit logical unit number. Instead, it uses an implicit internal logical unit number and the FOR_READ environment variable. If FOR_READ is not defined, the code READ (*,f) iolist or READ f,iolist reads from CONIN$ (standard input). If FOR_READ is defined (as a filename optionally containing a path), the specified file would be read.
FOR_NOERROR_DIALOGS
FOR_PRINT
FOR_READ
Page 33 of 34 If set to true, allows the continuation of the executing program when /check:flawed_pentium (default) is in effect and a flawed Pentium chip is detected. For more information, see Intel Pentium Floating-Point Flaw. The TYPE statement does not include an explicit logical unit number. Instead, it uses an implicit internal logical unit number and the FOR_TYPE environment variable. If FOR_TYPE is not defined, the code TYPE f,iolist writes to CONOUT$ (standard output). If FOR_TYPE is defined (as a filename optionally containing a path), the specified file would be written to. Lets you specify the file name for a particular unit number (n), when a file name is not specified in the OPEN statement or an implicit OPEN is used, and the compiler option /fpscomp:filesfromcmd was not specified. Preconnected files attached to units 0, 5, and 6 are by default associated with system standard I/O files. Lets you specify the data format for an unformatted file associated with a particular unit number (n), as described in Methods of Specifying the Data Format. Lets you specify the data format for unformatted files with a particular file extension suffix (ext), as described in Methods of Specifying the Data Format.
FOR_TYPE
FORTn
FORT_CONVERTn
FORT_CONVERT.ext FORT_CONVERT_ext
For example:
SET FOR_GENERATE_DEBUG_EXCEPTION=TRUE
Page 34 of 34
To view the current value of an environment variable, use the SET command with only the environment variable name:
SET FOR_GENERATE_DEBUG_EXCEPTION FOR_GENERATE_DEBUG_EXCEPTION=TRUE
From within your program, you can set the appropriate environment variable by calling the SETENVQQ routine:
program ENVVAR use dflib LOGICAL(4) res ! Add other data declarations here ! call SETENVQQ as a function res=SETENVQQ("FOR_GENERATE_DEBUG_EXCEPTION=T") ...
For a list of environment variables used with the DF command, see Environment Variables Used with the DF Command.
Page 1 of 22
l l
Representing Numbers Loss of Precision Errors: Rounding, Special Values, Underflow, and Overflow Setting and Retrieving Floating-Point Status and Control Words (ia32 systems only) Handling Arithmetic Exceptions Intel Pentium Floating-Point Flaw (ia32 systems only)
When the term floating-point unit (FPU) appears, it refers to your math processor. The descriptions in this section minimize hardware-specific terminology.
Representing Numbers
Fortran's numeric environment is flexible, which helps make Fortran a strong language for intensive numerical calculations. The Fortran standard purposely leaves the precision of numeric quantities and the method of rounding numeric results unspecified. This allows Fortran to operate efficiently for diverse applications on diverse systems. The effect of math computations on integers is straightforward:
l
Integers of KIND=4 consist of a maximum positive integer (2,147,483,647), a minimum negative integer (-2,147,483,648), and all integers between them including zero. Integers of KIND=8 consist of a maximum positive integer (-9,223,372,036,854,775,808), a minimum negative integer (9,223,372,036,854,775,807), and all integers between them including zero.
Operations on integers result in other integers within this range. The only arithmetic rule to remember is that integer division results in truncation (for example, 8/3 evaluates to 2). Computations on real numbers, however, may not yield what you expect. This happens because the hardware must represent numbers in a finite number of bits. There are several effects of using finite floating-point numbers. The hardware is not able to represent every real number exactly, but must approximate exact
Page 2 of 22
representations by rounding or truncating to finite length. In addition, some numbers lie outside the range of representation of the maximum and minimum exponents and can result in calculations that underflow and overflow. As an example of one consequence, finite precision produces many numbers that, although non-zero, behave in addition as zero. You can minimize the effects of finite representation with programming techniques; for example, by not using floating-point numbers in LOGICAL comparisons or by giving them a tolerance (for example, IF (x <= 10.001)), and by not attempting to combine or compare numbers that differ by more than the number of significant bits. (For more information on programming methods to reduce the effects of imprecision, see Rounding Errors.) For further discussion of how floating-point numbers are represented, see:
l l l
Floating-Point Numbers Retrieving Parameters of Numeric Representations Native IEEE Floating-Point Representations
Floating-Point Numbers
This version of Visual Fortran uses a close approximation to the IEEE floatingpoint standard (ANSI/IEEE Std 754-1985, IEEE Standard for Binary FloatingPoint Arithmetic, 1985). This standard is common to many microcomputerbased systems due to the availability of fast math coprocessors that implement the required characteristics. You should choose the appropriate setting of the /fpe compiler option to select the type of default floating-point exception handling provided by the Visual Fortran run-time system. This section outlines the characteristics of the standard and its implementation for Visual Fortran. Except as noted, the description includes both the IEEE standard and the Visual Fortran implementation. The following topics are discussed:
l l l l
Floating-Point Formats Floating-Point Representation Viewing Floating-Point Representations with BitViewer Special Values (Signed Zero, NaN, Signed Infinity)
Floating-Point Formats
The IEEE Standard 754 specifies values and requirements for floating-point representation (such as base 2). The standard outlines requirements for two formats: basic and extended, and for two word-lengths within each format:
Page 3 of 22
Visual Fortran supports single-precision format (REAL(4)) and double-precision format (REAL(8)) floating-point numbers. Visual Fortran sets the process control word by default to use double-precision run-time intermediate calculations. At some levels of optimization, some single-precision numbers are stored on the floating-point stack (which defaults to double precision) rather than being stored back into memory where they would be truncated to single precision. The compiler option /fltconsistency can control floating-point consistency and request that results be stored in memory rather than on the floating-point stack.
Floating-Point Representation
Floating-point numbers approximate real numbers with a finite number of bits. You can see the bits representing a floating-point number with the BitViewer tool. The bits are calculated as shown in the following formula. The representation is binary, so the base is 2. The bits bn represent binary digits (0 or 1). The precision P is the number of bits in the nonexponential part of the number (the significand), and E is the exponent. With these parameters, binary floating-point numbers approximate real numbers with the values: ( - 1)s b0 . b 1 b2 ... b
P-1
x 2E
where s is 0 or 1 (+ or - ), and Emin<= E <= Emax The following table gives the standard values for these parameters for single, double, and extended-double formats and the resulting bit widths for the sign, the exponent, and the full number. Parameters for IEEE Floating-Point Formats Parameter Sign width in bits P Emax Emin Exponent bias Exponent width in bits Format width in bits Single 1 24 +127 - 126 +127 8 32 Double 1 53 +1023 - 1022 +1023 11 64 Extended Double 1 64 +16383 - 16382 +16383 15 80
Page 4 of 22
The standard requires that the single and double formats be normalized, so b0 is always 1. The actual number of bits needed to represent the precisions 24 and 53 is therefore 23 and 52, respectively, because b0 is chosen to be 1 implicitly. Extended-double format need not be normalized, so it uses the full 64 bits for precision. A bias is added to all exponents so that only positive integer exponents occur. This expedites comparisons of exponent values. The stored exponent is actually: e = E + bias For more information:
l
On the floating-point representation, see Native IEEE Floating-Point Representations. On using the BitViewer tool, see Viewing Floating-Point Representations with BitViewer. On rReading or writing floating-point data other than native IEEE little endian data, see Converting Unformatted Numeric Data.
To view floating-point numbers in BitViewer, open the Data Type menu, then choose Floating-Point Real (or use the F9 shortcut key). Set the precision by
Page 5 of 22
selecting one of the choices in the Bytes box. Four bytes, REAL(4), displays the number in single format (23-bit precision). Eight bytes, REAL(8), displays the number in double format (52-bit precision). The following figures show the BitViewer display of the memory storage for a 4-byte real number and an 8-byte real number, both equal to 12.6. In the double format display, the most significant part is on the bottom and the least significant 32 bits above. Single Format in BitViewer
Note: BitViewer lets you view and manipulate integer and character data as well as floating-point, and to translate between different data types. Refer to the BitViewer online Help file (Help menu) for more information.
Special Values
Special cases of the exponent-significand combination represent four types of special values in addition to the normalized numbers. The following table shows all five types of values.
Page 6 of 22
IEEE Floating-Point Values Name Signed zero Denormalized number Normalized number Signed infinity Not a Number Quantity 0 0 . sig x 2Emin 1 . sig x 2E infinity NaN Exponent E = Emin - 1 E = Emin - 1 Emin<= E<= E E = Emax + 1 E = Emax + 1
max
Significand sig = 0 sig not equal 0 sig sig = 0 sig not equal 0
Signed zero Visual Fortran treats zero as signed by default. The sign of zero is the same as the sign of a nonzero number. If you use the intrinsic function SIGN with zero as the second argument, the sign of the zero will be transferred. Comparisons, however, consider +0 to be equal to -0. A signed zero is useful in certain numerical analysis algorithms, but in most applications the sign of zero is invisible.
Denormalized numbers Denormalized numbers (denormals) fill the gap between the smallest positive number and the smallest negative number. Otherwise only () 0 occurs in that interval. Denormalized numbers permit gradual underflow for intermediate results calculated internally in extended-double format. A status flag (on ia32 systems, the precision bit in the FPU Status Word exception field) is set when a number loses precision due to denormalization.
Signed infinity Infinities are the result of arithmetic in the limiting case of operands with arbitrarily large magnitude. They provide a way to continue when an overflow occurs. The sign of an infinity is simply the sign you obtain for a finite number in the same operation as the finite number approaches an infinite value. By retrieving the status flags described in Setting and Retrieving Floating-Point Status and Control Words in this section, you can differentiate between an infinity that results from an overflow and one that
Page 7 of 22
results from division by zero. Visual Fortran treats infinity as signed by default. The output value of infinity is Infinity or -Infinity.
l
Not a Number Not a Number (NaN) results from an operation involving one or more invalid operands. For instance 0/0 and SQRT(-1) result in NaN. In general, an operation involving a NaN produces another NaN. Because the fraction of a NaN is unspecified, there are many possible NaNs. Visual Fortran treats all NaNs identically, but provide two different types:
Signaling NAN, which has an initial fraction bit of 0 (zero). Quiet NaN, which has an initial fraction bit of 1.
EPSILON
EXPONENT
FRACTION
The Floating-Point Environment HUGE HUGE(x). Returns largest number that can be represented by data of type x. MAXEXPONENT(x). Returns the largest positive decimal exponent for data of the same type as x. MINEXPONENT(x). Returns the largest negative decimal exponent for data of the same type as x. NEAREST(x, s). Returns the nearest different machine representable number to x in the direction of the sign of s. PRECISION(x). Returns the number of significant digits for data of the same type as x. RADIX(x). Returns the base for data of the same type as x. RANGE(x). Returns the decimal exponent range for data of the same type as x. RRSPACING(x). Returns the reciprocal of the relative spacing of numbers near x. SCALE(x, i). Multiplies x by 2 raised to the power of i. SET_EXPONENT(x,i). Returns a number whose fractional part is x and whose exponential part is i. SPACING(x). Returns the absolute spacing of numbers near x. TINY(x). Returns smallest positive number that can be represented by data of type x.
Page 8 of 22 x: Integer or Real result: same type as x x: Real result: INTEGER(4) x: Real result: INTEGER(4) x: Real s: Real and not zero result: same type as x x: Real or Complex result: INTEGER(4) x: Integer or Real result: INTEGER(4) x: Integer, Real or Complex result: INTEGER(4) x: Real result: same type as x x: Real i: Integer result: same type as x x: Real i: Integer result: same type as x x: Real result: same type as x x: Real result: same type as x
MAXEXPONENT
MINEXPONENT
NEAREST
PRECISION
RADIX RANGE
RRSPACING
SCALE
SET_EXPONENT
SPACING TINY
Page 9 of 22
Underflow occurs when an arithmetic result is too small for the math processor to handle. Depending on the setting of the /fpe compiler option, underflows are set to zero (they are usually harmless) or they are left as is (denormalized). Overflow occurs when an arithmetic result is too large for the math processor to handle. Overflows are more serious than underflows, and may indicate an error in the formulation of a problem (for example, unintended exponention of a large number by a large number). Overflows produce an appropriately signed infinity value.
Inexact numbers, special values, underflows, and overflows are floating-point exceptions. You can select how rounding is done and how exceptions are handled by setting the floating-point control word. Setting the control word is described in Setting and Retrieving Floating-Point Status and Control Words (ia32 only) and exception handling in Handling Floating-Point Exceptions (ia32 only). For a further discussion of rounding errors see:
l
Rounding Errors
Page 10 of 22
Rounding Errors
Although the rounding error for one real number might be acceptably small in your calculations, at least two problems arise because of it. If you test for exact equality between what you consider to be two exact numbers, the rounding error of either or both floating-point representations of those numbers may prevent a successful comparison and produce spurious results. Also, when you calculate with floating-point numbers the rounding errors may accumulate to a meaningful loss of numerical significance. Carefully consider the numerics of your solution to minimize rounding errors or their effects. You might benefit from using double-precision arithmetic or restructuring your algorithm, or both. For instance, if your calculations involve arrays of linear data items, you might reduce the loss of numerical significance by subtracting the mean value of each array from each array element and by normalizing each element of such an array to the standard deviation of the array elements. The following code segment can execute differently on different systems and produce different results for n, x, and s. It also produces different results if you use the /fltconsistency (ia32 systems only) or /nofltconsistency compiler options. Rounding error accumulates in x because the floating-point representation of 0.2 is inexact, then accumulates in s, and affects the final value for n:
INTEGER n REAL s, x n = 0 s = 0. x = 0. 1 n = n + 1 x = x + 0.2 s = s + x IF ( x .LE. 10. ) GOTO 1 ! Will you get 51 cycles? WRITE(*,*) 'n = ', n, '; x = ', x, '; s = ', s
This example illustrates a common coding problem: carrying a floating-point variable through many successive cycles and then using it to perform an IF test. This process is common in numerical integration. There are several remedies. You can compute x and s as multiples of an integer index, for example, replacing the statement that increments x with x = n * 0.2 to avoid round-off accumulation. You might test for completion on the integer index, such as IF (n <= 50) GOTO 1, or use a DO loop, such as DO n= 1,51. If you must test on the real variable that is being cycled, use a realistic tolerance, such as IF (x <= 10.001). Floating-point arithmetic does not always obey the standard rules of algebra exactly. Addition is not precisely associative when round-off errors are
Page 11 of 22
considered. You can use parentheses to express the exact evaluation you require to compute a correct, accurate answer. This is recommended when you specify optimization for your generated code, since associativity may otherwise be unpredictable. The expressions (x + y) + z and x + (y + z) can give unexpected results in some cases, as the Visual Fortran Sample ASSOCN.F90 in the ...\DF98 \SAMPLES\TUTORIAL folder shows. This example demonstrates the danger of combining two numbers whose values differ by more than the number of significant digits. The Sample INTERVAL.F90 in the ...\DF98\SAMPLES\TUTORIAL folder shows how changing the rounding precision and rounding mode in the floating-point control word between calculations affects the calculated result of the following simple expression:
(q*r + s*t) / (u + v)
The Visual Fortran Sample EPSILON.F90 in the ...\DF98\SAMPLES\TUTORIAL folder illustrates difficulties that rounding errors can cause in expressions like 1.0 + eps, where eps is just significant compared to 1.0. The compiler uses the default rounding mode (round-to-nearest) during compilation. The compiler performs more compile-time operations that eliminate run-time operations as the optimization level increases. If you set rounding mode to a different setting (other than round-to-nearest), that rounding mode is used only if that computation is performed at run-time. For example, the Sample INTERVAL.F90 is compiled at /optimize:0, which disables certain compile-time optimizations, including constant propagation and inlining. For more information:
l
Page 12 of 22
discussions of rounding errors because it expresses the smallest floating-point number that you can add to 1.0 with a result that does not round to 1.0. Additional guard bits are included in floating-point hardware to allow rounding on the order of EPS. The result of any one floating-point operation is therefore tolerably imprecise, but the total error that results from many such operations on propagated numbers accumulates unavoidably.
Setting and Retrieving Floating-Point Status and Control Words (ia32 only)
The FPU (floating-point unit) on ia32 systems contains eight floating-point registers the system uses for numeric calculations, for status and control words, and for error pointers. You normally need to consider only the status and control words, and then only when customizing your floating-point environment. The FPU status and control words correspond to 16-bit registers whose bits hold the value of a state of the FPU or control its operation. Visual Fortran defines a set of symbolic constants to set and reset the proper bits in the status and control words. For example:
USE DFLIB CALL SETCONTROLFPQQ(FPCW$OVERFLOW .AND. FPCW$CHOP) ! set the floating-point control word to allow overflows ! and to round by truncation
The status and control symbolic constants (such as FPCW$OVERFLOW and FPCW$CHOP in the preceding example) are defined as INTEGER(2) parameters in the module DFLIB.F90 in the ...\DF98\INCLUDE folder. The status and control words are made of logical combinations (such as with .AND.) of different parameters for different FPU options. The name of a symbolic constant takes the general form name$option. The prefix name is one of the following: Prefixes for Parameter Flags name FPSW FPCW SIG FPE Meaning Floating-point status word Floating-point control word Signal Floating-point exception
Page 13 of 22
The suffix option is one of the options available for that name. The parameter name$option corresponds either to a status or control option (for example, FPSW$ZERODIVIDE, a status word parameter that shows whether a zero-divide exception has occurred or not) or name$option corresponds to a mask, which sets all symbolic constants to 1 for all the options of name. You can use the masks in logical functions (such as IAND, IOR, and NOT) to set or to clear all options for the specified name. The following sections define the options and illustrate their use with examples. You can control the floating-point processor options (on ia32 systems) and find out its status with the run-time library routines GETSTATUSFPQQ (ia32 only), GETCONTROLFPQQ (ia32 only), SETCONTROLFPQQ (ia32 only), and MATHERRQQ (ia32 only). Examples of using these routines also appear in the following sections. For more information:
l l
Floating-Point Status Word (ia32 only) Floating-Point Control Word (ia32 only)
The Floating-Point Environment FPSW$ZERODIVIDE FPSW$OVERFLOW FPSW$UNDERFLOW FPSW$INEXACT #0004 #0008 #0010 #0020 A divide by zero occurred An overflow occurred An underflow occurred Inexact precision occurred
Page 14 of 22
You can find out which exceptions have occurred by retrieving the status word and comparing it to the exception parameters. For example:
USE DFLIB INTEGER(2) status CALL GETSTATUSFPQQ(status) IF ((status .AND. FPSW$INEXACT) > 0) THEN WRITE (*, *) "Inexact precision has occurred" ELSE IF ((status .AND. FPSW$DENORMAL) > 0) THEN WRITE (*, *) "Denormal occurred" END IF
To clear the status word flags, call the CLEARSTATUSFPQQ (ia32 only) routine.
The Floating-Point Environment FPCW$24 FPCW$MCW_RC FPCW$CHOP FPCW$UP FPCW$DOWN FPCW$NEAR FPCW$MCW_EM FPCW$INVALID FPCW$DENORMAL FPCW$ZERODIVIDE FPCW$OVERFLOW FPCW$UNDERFLOW FPCW$INEXACT #0000 #0C00 #0C00 #0800 #0400 #0000 #003F #0001 #0002 #0004 #0008 #0010 #0020 24-bit precision Rounding control mask Truncate Round up Round down Round to nearest Exception mask Allow invalid numbers
Page 15 of 22
Allow denormals (very small numbers) Allow divide by zero Allow overflow Allow underflow Allow inexact precision
53-bit precision Round to nearest (rounding mode) The denormal, underflow, overflow, invalid, and inexact precision exceptions are disabled (do not generate an exception). To change exception handling, you can use the /fpe compiler option or the FOR_SET_FPE routine.
Exception Parameters
An exception is disabled if its bit is set to 1 and enabled if its bit is cleared to 0. If an exception is disabled (exceptions can be disabled by setting the flags to 1 with SETCONTROLFPQQ (ia32 only)), it will not generate an interrupt signal if it occurs. The floating-point process will return an appropriate special value (for example, NaN or signed infinity), but the program continues. You can find out
Page 16 of 22
which exceptions (if any) occurred by calling GETSTATUSFPQQ (ia32 only). If errors on floating-point exceptions are enabled (by clearing the flags to 0 with SETCONTROLFPQQ (ia32 only)), the operating system generates an interrupt when the exception occurs. By default these interrupts cause run-time errors, but you can capture the interrupts with SIGNALQQ and branch to your own error-handling routines. You should remember not to clear all existing settings when changing one. The values you want to change should be combined with the existing control word in an inclusive-OR operation (OR, IOR, .OR.) if you do not want to reset all options. For example:
USE DFLIB INTEGER(2) control, newcontrol CALL GETCONTROLFPQQ(control) newcontrol = (control .OR. FPCW$INVALID) ! Invalid exception set (disabled). CALL SETCONTROLFPQQ(newcontrol)
Precision Parameters
On ia32 systems, the precision bits control the precision to which the FPU rounds floating-point numbers. For example:
USE DFLIB INTEGER(2) control, holdcontrol, newcontrol CALL GETCONTROLFPQQ(control) ! Clear any existing precision flags. holdcontrol = (control .AND. (.NOT. FPCW$MCW_PC)) newcontrol = holdcontrol .OR. FPCW$64 ! Set precision to 64 bits. CALL SETCONTROLFPQQ(newcontrol)
The precision options are mutually exclusive. If you set more than one, you may get an invalid mode or a mode other than the one you want. Therefore, you should clear the precision bits before setting a new precision mode.
Rounding Parameters
On ia32 systems, the rounding flags control the method of rounding that the FPU uses. For example:
USE DFLIB INTEGER(2) control, clearcontrol, newcontrol CALL GETCONTROLFPQQ(control) ! Clear any existing rounding flags. clearcontrol = (control .AND. (.NOT. FPCW$MCW_RC)) newcontrol = clearcontrol .OR. FPCW$UP ! Set rounding mode to round up. CALL SETCONTROLFPQQ(newcontrol)
The rounding options are mutually exclusive. If you set more than one, you may
Page 17 of 22
get an invalid mode or a mode other than the one you want. Therefore, you should clear the rounding bits before setting a new rounding mode.
If a floating-point exception interrupt occurs and you do not have an exception handling routine, the run-time system will respond to the interrupt according to the behavior selected by the compiler options /fpe and /math_library. Remember, interrupts only occur if an exception is enabled (set to 0). If you do not want the default system exception handling, you need to write your own interrupt handling routine:
Page 18 of 22
Write a function that performs whatever special behavior you require on the interrupt. Register that function as the procedure to be called on that interrupt with SIGNALQQ.
Note that your interrupt handling routine must use the ATTRIBUTES C directive. The drawback of writing your own routine is that your exception-handling routine cannot return to the process that caused the exception. This is because when your exception-handling routine is called, the floating-point processor is in an error condition, and if your routine returns, the processor is in the same state, which will cause a system termination. Your exception-handling routine can therefore either branch to another separate program unit or exit (after saving your program state and printing an appropriate message). You cannot return to a different statement in the program unit that caused the exceptionhandling routine, because a global GOTO does not exist, and you cannot reset the status word in the floating-point processor. If you need to know when exceptions occur and also must continue if they do, you must disable exceptions so they do not cause an interrupt, then poll the floating-point status word at intervals with GETSTATUSFPQQ (ia32 only) to see if any exceptions occurred. To clear the status word flags, call the CLEARSTATUSFPQQ (ia32 only) routine. Obviously, polling the floating-point status word at intervals creates processing overhead for your program. In general, you will want to allow the program to terminate if there is an exception. An example of an exception-handling routine follows. The exception-handling routine hand_fpe and the program that invokes it are both contained in SIGTEST.F90 in the Visual Fortran Samples folder ...\DF98\SAMPLES\TUTORIAL. The comments at the beginning of the SIGTEST.F90 file describe how to compile this example.
! SIGTEST.F90 !Establish the name of the exception handler as the ! function to be invoked if an exception happens. ! The exception handler hand_fpe is attached below. USE DFLIB INTERFACE FUNCTION hand_fpe (sigid, except) !DEC$ ATTRIBUTES C :: hand_fpe INTEGER(4) hand_fpe INTEGER(2) sigid, except END FUNCTION END INTERFACE INTEGER(4) iret REAL(4) r1, r2 r1 = 0.0 iret = SIGNALQQ(SIG$FPE, hand_fpe) WRITE(*,*) 'Set exception handler. Return = ', iret ! Cause divide by zero exception
Page 19 of 22
The Floating-Point Environment MTH$E_OVERFLOW MTH$E_UNDERFLOW MTH$E_TLOSS MTH$E_PLOSS 3 4 5 6 Overflow range error Underflow range error Total loss of precision Partial loss of precision
Page 20 of 22
A domain error means that an argument is outside the math function's domain, for example, SQRT(-1). A singularity error means that an argument is a singularity value for the math function, and the result is not defined for that value, for example, LOG10(0.0). Overflow and underflow errors are the same as floating-point counterparts, and precision loss the same as floating-point inexact results. You can write a MATHERRQQ subroutine that resolves errors generated by math functions. Your MATHERRQQ can issue a warning, assign a default value if an error occurs, or take other action. If you do not provide your own MATHERRQQ subroutine, a default MATHERRQQ provided with the floatingpoint library will terminate the program. The following gives an example of an alternative MATHERRQQ subroutine (in MATHERR.F90 in the .../DF98/SAMPLES/TUTORIAL folder):
SUBROUTINE MATHERRQQ( name, length, info, retcode) USE DFLIB INTEGER(2) length, retcode CHARACTER(length) name RECORD /MTH$E_INFO/ info PRINT *, "Entered MATHERRQQ" PRINT *, "Failing function is: ", name PRINT *, "Error type is: ", info.errcode IF ((info.ftype == TY$REAL4 ).OR.(info.ftype == TY$REAL8)) THEN PRINT *, "Type: REAL" PRINT *, "Enter the desired function result: " READ(*,*) info.r8res retcode = 1 ELSE IF ((info.ftype == TY$CMPLX8 ).OR.(info.ftype == TY$CMPLX16)) THEN PRINT *, "Type: COMPLEX" PRINT *, "Enter the desired function result: " READ(*,*) info.c16res retcode = 1 END IF END
The following is a Visual Fortran Sample program (MATHTEST.F90 in the .../DF98/SAMPLES/TUTORIAL folder) that causes MATHERQQ to be called:
REAL(4) r1, r2 /-1.0/ REAL(8) r3, r4 /-1.0/ COMPLEX(4) c1, c2 /(0.0, 0.0)/ r1 = LOG(r2) r3 = SQRT(r4) c1 = CLOG(c2)
Page 21 of 22
If you compile and run this program without any compiler options (the default is /check:flawed_pentium), a run-time error occurs when it is run on a flawed Pentium system. Your operating system can also work around flawed Pentium processors by using software emulation for floating-point operations. Refer to your operating system documentation for more information. If the operating system has been
Page 22 of 22
configured for software emulation, then all floating-point operations in Visual Fortran will always operate correctly, including the above program. Note that the performance cost of an operating system fix can be very high, and if your program is run on another machine without the same operating system fix, it will execute incorrectly. If you distribute software that is susceptible to the floating-point problems of a flawed Pentium, and want your program to halt if it is run on a system with such a processor, you can check the processor when your application starts. To do this, convert the program above into a simple subroutine, call the subroutine at the start of your application, and use the STOP statement to stop the application before it begins if a flawed Pentium processor is detected. If you distribute your software, you should compile it with the /check:flawed_pentium compiler option. All the run-time libraries that come with Visual Fortran have been compiled to be safe with respect to the Pentium divide and MOD problems. For more information on the Intel Pentium flaw, or to request a replacement Pentium processor, you can contact Intel in the United States at 1-800-6288686.
Page 1 of 24
l l l
Default Exception Handling, Console Event Handling, and Termination Handling Structure of a Visual Fortran Application When to Provide Your Own Exception/Termination Handler How to Provide Your Own Exception/Termination Handler
To fully appreciate this discussion, you will need knowledge of Windows structured exception handling (SEH). A good reference on this subject is Chapter 16 in the book Advanced Windows (Third Edition) by Jeffrey Richter. To employ some of the exception handling techniques presented, you will need a C language compiler, such as Microsoft's Visual C++ product (part of Visual Studio), which has support for try-except constructs or some other form of support for structured exception handling.
General Default Exception Handling How the Floating-Point Exception Handling (/fpe) Compiler Option Works Default Console Event Handling General Default Termination Handling
A default exception handler is included with Fortran Console, Fortran QuickWin, and Fortran Standard Graphics applications. No default exception handler is included with Fortran Windows or Fortran DLL applications.
Page 2 of 24
Most exceptions captured by the Visual Fortran default handler are dealt with as severe errors. When an exception occurs, the Fortran run-time system will display an error message and traceback output as described in Handling RunTime Errors. A run-time error with a severe status causes the Visual Fortran run-time system to terminate the application. Most I/O programming errors are also severe and will terminate an application. I/O programming errors are not exceptions and cannot be caught by an exception handler. An unhandled I/O programming error is reported through a different mechanism in the Visual Fortran run-time system. Regardless of the application (project) type, unhandled I/O programming error will generate an error message and traceback output.
Using /fpe:3, the default on ia32 systems, implies full IEEE support. There are no floating-point traps in this mode, although floating-point exceptions are detected and recorded in the floating-point status word. The second option, /fpe:0, enables trapping on floating-point exceptions and flushing of underflow results to 0.0.
The full consequences of the /fpe compiler option depend upon the type of application you are building. You only get the full support for the chosen behavior in a Fortran Console or QuickWin/Standard Graphics application, assuming you do not override the default run-time exception handler. The work to achieve the full behavior is done partly by each of the default run-time handler, the Fortran compiler, the math library, the underlying hardware, and the operating system. The following sections discuss:
l
l l
Floating-Point Exceptions in Fortran Console, Fortran QuickWin, and Fortran Standard Graphics Applications Floating-Point Exceptions in Fortran DLL Applications Floating-Point Exceptions in Fortran Windows Applications
Page 3 of 24
Floating-Point Exceptions in Fortran Console, Fortran QuickWin, and Fortran Standard Graphics Applications When you build a console application, the compiler generates a few calls at the beginning of your Fortran main program to Fortran run-time routines that initialize the environment, either with default options or in accordance with your selected compile-time options.
l
For floating-point exception handling, /fpe:3 is the default on ia32 platforms. The run-time system initializes the hardware to mask all exceptions. With /fpe:3: The Visual Fortran run-time system does this initialization automatically (no call from the compiled code). The ia32 hardware automatically generates the default IEEE result for exceptional results. Because traps are masked with /fpe:3, there are no traps and you see exceptional values like Nan's, Infinities, and denormalized numbers in your computations. Users can poll the floating-point status word with GETSTATUSFPQQ to see if an exception has occurred and can clear the status register with CLEARSTATUSFPQQ.
If you specify /fpe:0, the compiler generates a call to a Visual Fortran runtime routine, FOR_SET_FPE, with an argument that unmasks all floatingpoint traps in the floating-point control word. In this case, the hardware does not supply the default IEEE result. It traps to the operating system, which then looks for a handler. In a Fortran console or Fortran QuickWin application, the Visual Fortran run-time system provides a default handler unless you establish your own. For all exceptions except underflow, the run-time system just prints out an error message and aborts the application. For underflow, the run-time system replaces the result with zero. This treatment of denormalized numbers with /fpe:0 is called abrupt underflow to 0 (zero), as opposed to gradual underflow to 0 provided with /fpe:3. There are no denormalized numbers with /fpe:0. Further, if you also select /check:underflow with /fpe:0, you get a message at run-time for each of the first two occurrences of a floating-point underflow and a total count of underflows when the application exits. The /check:underflow option is not supported with /fpe:3.
Fixing up underflow results to zero can significantly degrade the performance of your ia32 application. If you are experiencing a large number of underflows, consider changing your code to avoid underflows or consider masking underflow
Page 4 of 24
traps and allowing the hardware to operate on denormalized numbers. The ia32 hardware is designed to operate correctly in the denormalized range and doing so is much faster than trapping to fix up a result to zero. Another important point to understand about selecting the /fpe option is that the generated code must support the trapping mode. When an ia32 floating-point instruction generates an exceptional result, you do not necessarily get a trap. The instruction must be followed by an fwait instruction or another floating-point operate instruction to cause the trap to occur. The Visual Fortran compiler generates machine code to support these requirements in accordance with your selected /fpe option setting. In other words, the generated code must support the trapping mode. You can see this by compiling a simple test program and look at the machine code listing with /fpe:0 first, then /fpe:3. There are no fwait instructions in the /fpe:3 code. Even if you replace the default run-time exception handler with your own handler, you may still want to compile with /fpe:0 to generate code that supports trapping. Floating-Point Exceptions in Fortran DLL Applications In a DLL, there is no main Fortran program (unless you have written your main program in Fortran), so there is no automatic calling of run-time routines to initialize the environment. Even if you select /fpe:0, there is nothing that causes the run-time system to unmask traps in the hardware so you won't see traps. You will continue to see the hardware generated default IEEE results (Nan's, Infinities, and and denormalized numbers in your computations). The generated code will still do its part by supplying the fwait instructions, and so on, but unless the traps are unmasked somehow, no traps will occur. You can use SETCONTROLFPQQ or FOR_SET_FPE to unmask traps in the floating-point control word. There is also no default exception handling in a DLL. The main application that calls the DLL must provide this, or the code in the DLL must provide something when it is called. Since underflow processing (fixup to 0, and so on) is done by the default Fortran run-time system handler, a DLL won't have that feature automatically. A typical strategy is to compile with /fpe:0, but only unmask floating divide by zero, floating overflow, and floating invalid traps in the floating-point control word. By leaving floating-point underflow traps masked, the hardware will continue to provide gradual underflow through the denormalized range, but other floating-point exceptions will generate traps, which the user then handles as desired. Floating-Point Exceptions in Fortran Windows Applications In a Fortran Windows application, the situation is similar to a Fortran DLL
Page 5 of 24
application. You define a WinMain routine in your Fortran code as the main entry point for your application. The Fortran run-time system does not define the main routine as it does in a Fortran console or Fortran QuickWin (or Standard Graphics) application. Your code is not protected by the default handler and the default run-time initialization routines are not called.
These event types will result in an orderly program abort with an appropriate diagnostic message. Other console events such as a CTRL_LOGOFF_EVENT or CTRL_SHUTDOWN_EVENT are not handled by the default handler. The handler is notified of these events but returns FALSE to the operating system. This allows a Visual Fortran application activated as a Windows service to continue execution when a user logs off.
l l
If the default exception handler is used and the compiler /check:underflow and /fpe:0 options were selected, a count of the total number of floating point underflows is reported. Any Open files are closed and the requested DISPOSITION operations are performed. With a QuickWin application, any open QuickWin windows are closed. The C run-time exit() routine is called with the status code to return to the operating system. The C run-time exit() routine will call the Win32 routine ExitProcess to terminate the process. (See crt0dat.c in the C run-time sources).
In a Fortran DLL or Fortran Windows application, any unhandled I/O programming errors will cause the following actions:
Page 6 of 24
Any Open files are closed and the requested DISPOSITION operations are performed. The C run-time exit() routine is called with the status code to return to the operating system. The C run-time exit() routine will call the Win32 API routine ExitProcess to terminate the process. (See crt0dat.c in the C runtime sources.)
Any unhandled exceptions that occur in a Fortran DLL or Fortran Windows application will have application dependent behavior. Since there is no Fortran default handler present, the behavior depends on what you provide for a handler. If you do not explicitly provide a handler, the default mechanisms provided in your main program will determine the behavior. In a Fortran Windows application, the C run-time system will terminate the application.
Console Applications QuickWin and Standard Graphics Applications DLL Applications Windows Applications
Fortran Console Applications Fortran Console applications actually look like C applications under the hood, with the Visual Fortran run-time system providing the C main() function. The entry point for a console application is specified as the C library's mainCRTStartup() routine (see module crt0.c in the C run-time sources). This initializes the C run-time system, wraps the Fortran run-time system main() in a try-except construct using the C run-time's exception filter (_XcptFilter()), and calls the Fortran run-time system main() routine in run-time module for_main.c. In simplified form, it looks like this:
mainCRTStartup() { C initialization code here __try { more initialization code here mainret = main() /* calls Fortran run-time main() */ exit(mainret) } __except ( _XcptFilter() ) { _exit ( GetExceptionCode() ) } }
Page 7 of 24
In the Fortran run-time system, main() initializes the Fortran run-time system (if not already initialized), wraps a try-except construct around MAIN__ (the entry point to the Fortran code) with a filter expression that invokes the Fortran run-time system default handler on exceptions, and calls MAIN__. It also wraps a try-finally construct around all of this so run-time system clean up gets done (with FOR_RTL_FINISH_) when the program exits. In simplified form, it looks like this:
main() { __try { __try { for_rtl_init() MAIN__ } __except ( expression-invoking-fortran-default-handler ) { } } __finally { for_rtl_finish() } }
In the Fortran code, symbol MAIN__ is the entry point called by the run-time system's main() routine. MAIN__ has the code to do any further run-time initialization or checks. For example, if the user compiled with the nondefault /fpe:0 option, there would be a call to FOR_SET_FPE to tell the run-time system how to setup/react to floating-point exceptions. Fortran QuickWin and Standard Graphics Applications A Fortran QuickWin (including Fortran Standard Graphics) application is a specialized windows application where Visual Fortran provides the WinMain() function. The entry point for a QuickWin application is specified as the C library WinMainCRTStartup() routine (see module crt0.c in the C run-time sources). This gets the C run-time initialized, wraps the Visual Fortran defined WinMain() in a try-except construct using the C run-time exception filter (_XcptFilter()) and calls the Visual Fortran defined WinMain() routine. In simplified form, it looks like this:
WinMainCRTStartup() { C initialization code here __try { more initialization code here mainret = WinMain() /* calls CVF qwin library WinMain() */ exit(mainret) } __except ( _XcptFilter() ) { _exit ( GetExceptionCode() ) } }
In the QuickWin library, WinMain() performs some initialization specific to QuickWin, creates a new thread which begins execution at QWINForkMain, and
Page 8 of 24
then sits in a message loop directing the action. The message loop is wrapped in a try-except-finally construct which invokes the Fortran run-time system default handler if an exception occurs, and calls FOR_RTL_FINISH_ at exit. QWINForkMain() running in the other thread calls the Fortran run-time system main(), which in turn calls MAIN__. In simplified form, it looks like this:
WinMain() { Initialization code here BeginThreadEx (..., QWINForkMain, ... ) __try { __try { the message loop... for_rtl_finish() return (msg.wParam) } __except ( expression-invoking-default-fortran-handler ) { } } __finally { for_rtl_finish() return (msg.wParam) } }
The routines main() and MAIN__ are the same as previously described for a Fortran Console application. Fortran DLL Applications A Fortran DLL is a collection of one or more routines that you generally call from some other main program. As such, the routines execute in the structure and environment created by the code which calls into the DLL. You can provide DLL initialization through a DllMain() function, but you probably would control general application initialization from the main program. There are no automatic provisions for any exception handler in a DLL. There is no environment initialization except what you provide. Of course, if your main application is also written in Fortran, you will get the default Fortran handlers provided by that application type. Fortran Windows Applications A Fortran Windows application has as its entry point WinMainCRTStartup() and each user writes the code for the WinMain function declaration and interface. Examples are provided to show how to do this in Fortran code. The compiler still
Page 9 of 24
generates symbol MAIN__ with the initialization code in place, but nothing calls MAIN__. Also, nothing connects up to the run-time system's main() so there's no try-except construct to hook in the default Visual Fortran handler, and no run-time system initialization or cleanup. In simplified form, it looks like this:
WinMainCRTStartup() { C initialization code __try { more initialization code mainret = WinMain() /* calls the user's WinMain() */ exit(mainret) } __except ( _XcptFilter() ) { _exit ( GetExceptionCode() ) } }
Suppose your application creates some files during the course of its execution and you do not want to leave them on the disk if an unexpected error or exception occurs. The default termination actions only cause the files to be closed if you specifically opened them with DISPOSE='DELETE'. But suppose you do not want them deleted under normal termination. If an unexpected event occurs, you need to get control so you can clean up these files as needed. Perhaps your application can recover from a particular situation, for example, an integer divide-by-zero operation. You want to gain control if that exception occurs and deal with it. Perhaps you just want to output an application-specific error message when an exception occurs. You are building a Fortran DLL to run under a Visual Basic GUI and you do
Page 10 of 24
not want the DLL to crash the application if an exception occurs in the DLL. Your code takes a lock on a global resource and you want to be sure and release the resource if an unexpected event occurs.
The list of possibilities is endless and only the application developer can anticipate his or her particular needs. See How to Provide Your Own Exception/Termination Handler for some ideas on how to deal with unexpected events when implementing a handler for your application.
Using Windows Structured Exception Handling (SEH) Using SIGNALQQ and How SIGNALQQ Works Suggestions for Console Event Handlers
Advanced Exception and Termination Handling Considerations Standard Graphics Applications Custom Handlers For Fortran DLL Applications Custom Handlers for Fortran Windows Applications
Page 11 of 24
l l
Custom Handlers for Fortran Console, Fortran QuickWin, and Fortran Standard Graphics Applications Fortran Console and Fortran QuickWin (and Fortran Standard Graphics) applications have the full benefit of the Fortran default exception and error handling processing facilities. You may, however, want to supplement or replace the default facilities. Two Visual Fortran samples show how you can accomplish this:
l l
Sample Program CSLEXCP2 The following discussion refers to the sample program CSLEXCP2 provided with the Visual Fortran kit in the ...\SAMPLES\EXCEPTIONHANDLING folder. Suppose you have the problem described earlier of deciding what to do about open files your application has created in the event something unexpected occurs during execution of your application. Sample application CSLEXCP2 shows one way you might approach solving the problem. Looking at CSLEXCP2.F90, there is a main program CSLEXCP2, and three additional routines, MY_CODE, MY_EXCEPTION_FILTER_CODE, and MY_TERMINATION_HANDLER_CODE. Routine MY_CODE is the main body of Fortran code in the application. It opens (OPEN statement) a data file and sits in a loop prompting for two integers, and writes (WRITE statement) the result of dividing the first by the second integer to the data file. If the user enters the sentinel value of 99 for variable I (and not 0 for J), the application exits normally and the data file is closed (CLOSE statement) by the Fortran run-time system as a result of the call to the EXIT subroutine, but not deleted. If the user enters a 0 (zero) for variable J, an integer divide by zero exception occurs. This is the code we want to protect with a custom exception handler. Look at the main program, CSLEXCP2:
PROGRAM CSLEXCP2 INTERFACE SUBROUTINE MY_CODE_WRAPPER() !DEC$ IF DEFINED(_X86_) !DEC$ATTRIBUTES C, ALIAS:'_MY_CODE_WRAPPER' :: MY_CODE_WRAPPER !DEC$ ELSE
Page 12 of 24
!DEC$ATTRIBUTES C, ALIAS:'MY_CODE_WRAPPER' :: MY_CODE_WRAPPER !DEC$ ENDIF END SUBROUTINE END INTERFACE CALL MY_CODE_WRAPPER() END
All CSLEXCP2 does is call a routine named MY_CODE_WRAPPER. MY_CODE_WRAPPER is a C routine that implements the custom handler. To see how this is done, look at file wrapper.c. MY_CODE_WRAPPER wraps a try-except construct around a call to the main body of the Fortran application, MY_CODE. It also wraps a try-finally construct around the try-except construct. If an exception occurs during execution of MY_CODE, the operating system will evaluate the except filter expression to see if this handler wishes to handle the event. The except filter expression is just a call to the Fortran routine MY_EXCEPTION_FILTER_CODE. If MY_EXCEPTION_FILTER_CODE returns EXCEPTION_EXECUTE_HANDLER, the operating system continues execution in the except handler block, which in this case is empty. Execution would then continue in the __finally block which calls the general termination handler for the application, MY_TERMINATION_HANDLER_CODE. Routine MY_EXCEPTION_FILTER_CODE queries the user to see if the data file should be deleted or not. That is all the basic code needed to handle the exception situation. Routine MY_TERMINATION_HANDLER_CODE just prints a message and calls the EXIT subroutine to exit the application. Note, if MY_EXCEPTION_FILTER_CODE returns EXCEPTION_CONTINUE_SEARCH, the operating system continues searching its list of exception handlers and finds the default Fortran run-time exception filter. The code in the __finally block would never be executed because the default run-time handler will exit the process as part of its default handling of the exception. You can experiment with this when you reply to the handler query "Enter A to exercise the termination handler code path,...or B to pass the exception to the default handler." There are a couple of points to notice in this example. First, remember that this is a console application and its infrastructure basically looks like a C program. The entry point for the application is mainCRTStartup in the C run-time system. Routine mainCRTStartup() initializes the C run-time system and calls the Fortran run-time system main() routine. The Fortran run-time main() routine initializes the Fortran run-time system and calls entry point _MAIN__ in the Fortran code. If you build the sample and request a listing with machine code from the Fortran compiler, you can see the bit of prolog code that the Fortran compiler generates which provides further initialization before actually executing your Fortran code. Here is an excerpt of that listing:
Page 13 of 24
55 EC8B 24EC83 53 000000E8 01 04EC83 0004058D 0002 50 000000E8 01 04C483 04EC83 00001D8D 0002 53 000000E8 01 04C483 000000E8
0000 0000 0001 0003 0006 0007 000C 000F 0015 0016 001B 001E 0021 0027 0028 002D 0030
.CODE PUBLIC _MAIN__ PROC push mov sub push call sub lea push call add sub lea push call add call
_MAIN__ ebp ebp, esp esp, 36 ebx _for_check_flawed_pentium@0 esp, 4 eax, dword ptr .literal$+4 eax _for_set_fpe_@4 esp, 4 esp, 4 ebx, dword ptr .literal$ ebx _for_set_reentrancy@4 esp, 4 _MY_CODE_WRAPPER
Notice the calls to the Fortran run-time routines _for_check_flawed_pentium@0, _for_set_fpe_@4, and _for_set_reentrancy@4 that occur prior to the call to _MY_CODE_WRAPPER. This code was compiled with /fpe:0 selected, so you can see the call to _for_set_fpe_@4 to set up the non-default floating-point behavior on ia32 systems. This call would not be there with the default /fpe:3 option. The sample does not do any floating-point calculation (the /fpe:0 option was chosen just to illustrate the point). Likewise, if you deselect the check for flawed Pentium run-time option, the call to _for_check_flawed_pentium@0 would not be generated. This prolog code is only executed as a result of the underlying structure of Fortran Console, Fortran QuickWin, and Fortran Standard Graphics applications. It would not be executed in a Fortran Windows application or Fortran DLL because there would be no Fortran run-time system main() routine present to call MAIN__. Most of the work of handling the exception is really done in the course of executing the filter expression. The same is true of the Fortran default handler. So, if you allow an exception to be passed to the default run-time system handler, you will not get control again because the default handler will exit the process from within its exception filter in almost all cases. This short circuits any exception unwinding so your termination handler (the __finally construct) will never execute. The default run-time exception handler plays a role in how an application behaves when exceptions occur while running under the debugger. Normally, the occurrence of an exception while debugging would cause the default handler
Page 14 of 24
to do its normal activities except it would not exit the process. Instead, it returns EXCEPTION_CONTINUE_SEARCH so the operating system will give the debugger a second chance at handling the exception and the debugger will stop in your code with the cursor pointing where the unhandled exception happened. If you are more interested in seeing where the exception happened than you are to see your termination handler execute, you can return EXCEPTION_CONTINUE_SEARCH from your exception filter and let the default run-time system handler get the application stopped in the debugger as usual. In this (CSLEXCP2) example, no traceback output is produced to show where the integer divide by zero happened. This is normally done by the default handler, but we have overridden that handler. The next example shows that you can generate the traceback output in your custom handler using the TRACEBACKQQ routine. Sample Program CSLEXCP4 The following discussion refers to the sample program CSLEXCP4 provided with the Visual Fortran kit in the ...\SAMPLES\EXCEPTIONHANDLING folder. Suppose you want to allow the default handler to handle some exceptions, for example, floating-point underflow with /fpe:0. You would like to have the default handler provide the result fix up to zero for this case. Also, suppose you want to turn an exception into a simple program status code which your code can then act upon. Sample CSLEXCP4 will demonstrate a few techniques to accomplish this. The premise of sample CSLEXCP4 is that a Fortran subroutine is to be called which may generate floating point exceptions. We would like to detect floating point divide by zero exceptions and take some useful action to shut down the application gracefully. The application is to be compiled /fpe:0 and it is desired to have the default Fortran run-time system handler continue handling floating point underflow exceptions. Looking at the code in CSLEXCP4.F90, the subroutine to be protected is called DIVIDE. In a real application, the protected subroutine could be any complex sequence you like, but in this sample, the subroutine simply does a divide of two input numbers and prints the result. The main Fortran program does not call DIVIDE directly. It makes the call through a C wrapper function called DIVIDE_WRAPPER. The DIVIDE_WRAPPER function, in file WRAPPER.C, wraps the call to DIVIDE in a try-except construct. In this simple example, it is assumed that if DIVIDE returns without an exception, then all is well and DIVIDE_WRAPPER sets the rtn_sts variable accordingly. If no exception occurs, the except filter expression is never evaluated and the except block is never executed. Execution continues with the return statement just below the except handler block that returns
Page 15 of 24
If an exception does occur during execution of DIVIDE, the except filter expression is evaluated by the operating system. The filter expression takes two actions: 1. It captures the exception code related to the event in a local variable, ecode, for use in the exception handler code block. If the exception handler code block is executed, it will examine the exception code to determine a status value to return in rtn_sts. 2. It calls a Fortran function called CHECK_EXCEPTION_INFO, passing it the value returned from the Win32 routine GetExceptionInformation. If function CHECK_EXCEPTION_INFO returns EXCEPTION_EXECUTE_HANDLER, the operating system will execute the except handler code block. The handler block will examine the exception code and set rtn_sts to a value indicating a divide by zero occurred or to a value indicating some other exception occurred. Execution then continues with the return statement that returns the value of rtn_sts. If function CHECK_EXCEPTION_INFO returns EXCEPTION_CONTINUE_SEARCH, the operating system will continue looking back up the list of exception filters looking for one that wants to handle the exception. In this sample, the next exception filter is the default Fortran run-time system exception filter. The value passed to CHECK_EXCEPTION_INFO is the address of a data structure which contains pointers to an operating system supplied exception record and context record. CHECK_EXCEPTION_INFO uses this information to determine what happened and how to proceed. If this was a floating-point divide by zero, CHECK_EXCEPTION_INFO calls TRACEBACKQQ to produce a stack trace along with an application specific message. It then sets the return value of the function to EXCEPTION_EXECUTE_HANDLER and returns. If this was a floating-point underflow exception, CHECK_EXCEPTION_INFO calls TRACEBACKQQ again, only so you can see what is happening in the application output. Then, it sets the return value of the function to EXCEPTION_CONTINUE_SEARCH. When the operating system sees this return value, it evaluates the Fortran run-time default except filter expression, which causes the floating underflow result to be fixed up with zero, and execution will continue in the main program at the point of the exception. The main program initializes two arrays with a set of three values that will demonstrate each path of execution. A DO loop is traversed for three iterations: 1. The first trip around the loop shows what happens when there are no exceptions. 2. The second iteration shows a floating underflow being flushed to zero in the
Page 16 of 24
default handler. 3. The third (final) iteration shows a floating divide by zero being processed. The program takes action to shutdown the application with a simple STOP statement and message. Custom Handlers For Fortran DLL Applications There are two aspects of creating custom handlers for Fortran DLL applications:
l l
Containing Errors and Exceptions in Fortran DLL's Enabling Floating-Point Traps in Fortran DLL's
Containing Errors and Exceptions in Fortran DLL's If you are building a Fortran DLL and intend to call it from a Visual Basic GUI or a main program written in some other language, you want to be careful that errors and exceptions in the DLL do not crash your main application. Here are a few basic principles to keep in mind if you are building a Fortran DLL:
l
Construct your library routines so that they return a status to the caller and let the caller decide what to do. In order to return an expected status to the caller, you need to be defensive in your library code, so consider these other principles:
Where it makes sense, have the library code check input arguments passed in from the caller to make sure they are valid for whatever the library routine is going to do with them. For example, suppose the routine implements some numerical algorithm that has a valid domain of inputs it can act on and still produce well defined behavior. You can check the input arguments before you execute the algorithm and avoid unexpected behavior that might otherwise result (like unexpected floating-point exceptions). You might use Fortran intrinsic procedures like ISNAN and FP_CLASS to detect exceptional IEEE numbers. Your DLL code needs to return a status to the caller indicating the problem and let the caller take the appropriate action (gracefully shut down the application, try again with different input, etc.). In your library code, always check the success or failure of calls to I/O routines and dynamic memory allocation/deallocation. In Fortran, the I/O statements have optional ERR, END, EOR, and IOSTAT arguments that you can use to determine if the I/O requested was successful. Dynamic memory ALLOCATE and DEALLOCATE statements have an optional STAT specifier that allows you to obtain the status of the dynamic memory allocation/deallocation and prevent program termination. If you do not specify an action to take on an error, the Fortran run-time
Page 17 of 24
system has no choice but to deal with the error as an unhandled severe error and terminate the program. For a specific example of using IOSTAT and ERR to deal gracefully with an OPEN statement that gets a file-notfound error, see Using the IOSTAT Specifier and Fortran Exit Codes. You can do the same sort of thing in your code, but just return the status back to your Visual Basic or other non-Fortran main program and let it decide what to do.
l
Try to write your DLL code so unexpected program exceptions cannot occur, but devise a strategy for dealing with unexpected exceptions if they do happen. The most effective alternative for dealing with an exception is to use Win32 Structured Exception Handling support to gain control when an exception happens. Wrap all your DLL routine calls in C try/except constructs and have the except() filter expression call a routine you define which determines how to respond.
Enabling Floating-Point Traps in Fortran DLL's Before you can worry about how you will handle a floating-point trap condition occurring in a DLL, you have to consider the problem of unmasking those traps so they can occur. If you are compiling /fpe:3 and polling the floating-point status word to check for exceptions, you do not have to worry about the problem of unmasking traps. You do not want traps unmasked in that case. However, if your strategy is to compile /fpe:0 and allow traps on floating-point exceptions, you need to take action to unmask the traps in the floating-point control word because most other languages mask traps by default. Recall that a Fortran Console or Fortran QuickWin (or Standard Graphics) application would have unmasked traps for you automatically because the Fortran run-time system provides the main program and calls your MAIN__ which executes some prolog code before the actual application code starts. You do not have that in a Fortran DLL called by some other language. Different languages establish different initial environments. You must provide the desired initial environment yourself. The samples discussed below and in the discussion of Fortran Windows applications give some ideas on how and when to do this. Sample Program VBVF1 The following discussion refers to the Visual Fortran sample program VBVF1 provided in the ...\SAMPLES\EXCEPTIONHANDLING folder. VBVF1 shows how you might incorporate structured exception handling and standard Fortran I/O error handling techniques to protect a Visual BASIC GUI from exceptions and errors occurring in a Fortran DLL called by the GUI. The error handling techniques are general in nature and could be used in other situations as well. For example, a C main program calling a Fortran DLL could also use the ideas presented here.
Page 18 of 24
Sample application VBVF1 attempts to contain the effects of unexpected errors and exceptions occurring in a Fortran DLL by turning the unexpected events into user defined status conditions which the Visual Basic code can deal with gracefully. The GUI has six buttons each of which generates a particular error condition in the Fortran DLL. A seventh button allows you to clear the error status displayed in the GUI. There are three text boxes in the GUI. One box displays a message associated with the last error. Another box displays an error number for the last error and the third box displays the Visual Basic source file where the error occurred. Two of the buttons, "Test Integer Divide By Zero Response" and "Test Floating Point Overflow Response," are coded to show the use of a Visual Basic handler to deal with an error occurring in the Fortran DLL. While Visual Basic error handling works for these two specific cases to some extent, it is not a very robust solution for dealing with the more general case. If you are not familiar with Visual Basic error handling, this simple example should begin to give you an idea of what can be done. Refer to Visual Basic documentation for more complete information. Each of the four remaining buttons causes a call to routine Fortran_Test in the Fortran DLL. See source file gen_errs.f90. Routine Fortran_Test generates the selected error. The Visual Basic code does not call Fortran_Test directly however. All Visual Basic calls to Fortran_Test go through a wrapper routine, Fortran_Test_Wrapper. Fortran_Test_Wrapper (source file wrappers.c) wraps a C try-except construct around calls into the DLL and acts as a control point for containing exceptions and errors that may unexpectedly occur in the DLL. Fortran_Test_Wrapper always returns a user defined application specific status to the Visual Basic calling routine. The Visual Basic code then reacts to the status value by displaying some appropriate message to the user. The implementation of this scheme depends on two things: 1. The solid programming practice of always requesting and checking I/O status in Fortran I/O statements 2. The use of the try-except construct in the wrapper routine. In the __try block of the wrapper routine, a call is made to Fortran_Test, which expects a Fortran defined status value to be returned. For the test cases that generate an I/O error, the code in Fortran_Test captures the I/O status and returns it to Fortran_Test_Wrapper. Fortran_Test_Wrapper then calls another DLL routine, TRANSLATE_TO_MY_APP_CODES, which does a table look up to translate the Fortran defined status code to a user defined status code. Execution continues with the return statement following the __except block. The Visual Basic code gets the user defined status value and puts up the appropriate
Page 19 of 24
If an exception occurs during the execution of Fortran_Test, control transfers to the operating system. The operating system examines its list of exception handlers and finds the exception filter associated with Fortran_Test_Wrapper. It evaluates the __except expression to see if the exception will be handled here. The __except expression calls another DLL routine, CHECK_EXCEPTION_INFO, passing it the operating system supplied information about the exception. CHECK_EXCEPTION_INFO determines what the exception condition was, generates traceback information with an appropriate message specific to the event and returns EXCEPTION_EXECUTE_HANDLER. Returning EXCEPTION_EXECUTE_HANDLER tells the operating system to resume execution in the handler block associated with the __except construct in Fortran_Test_Wrapper. The code in the handler block calls TRANSLATE_TO_MY_APP_CODES, which again does a table look up, but to a second table, to translate the Windows defined status code to a user defined status code. Execution continues with the return statement following the __except block. The Visual Basic code gets the user defined status value and puts up the appropriate display. In this sample, the Fortran code is compiled /fpe:0 and intentionally creates floating point exceptions. It expects traps due to those exceptions. By default, Visual Basic masks all floating point traps so the application must take action to unmask traps. For this sample, two routines are provided in the DLL, Fortran_FP_Trap_Enable and Fortran_FP_Trap_Disable, that are called from the Visual Basic code. In procedure cmdFltOvf_Click, source file frmVBVF1.frm, there is a call to DLL routine Fortran_FP_Trap_Enable to unmask traps before the call to DLL routine Fortran_FltOvf which generates the exception. Fortran_FP_Trap_Enable uses GETCONTROLFPQQ and SETCONTROLFPQQ to unmask traps in the floating-point control word. In the Visual Basic handler code for this routine at label My_VB_FltOvf_Handler, DLL routine Fortran_FP_Trap_Disable is called to revert to Visual Basic default exception settings. The timing of when and how to unmask or mask floating-point traps is an application design issue. The sample just shows one way you can do it. Custom Handlers for Fortran Windows Applications Fortran Windows applications are not hooked up to the Fortran default exception handling processing facilities. Fortran Windows applications are considered to be an area devoted to full customization, and the Fortran run-time system tries to "stay out of the way," so you can do whatever you want in your code. The following discussion refers to the sample program WINEXCP1 provided with the Visual Fortran kit in the ...\SAMPLES\EXCEPTIONHANDLING folder. WINEXCP1 extends the GENERIC sample to show how you might incorporate structured
Page 20 of 24
exception handling and standard I/O error handling techniques to protect a Fortran message processing routine. If you build and run this application, the main window will be displayed and you will see a menu labeled "Run Error Simulator." On the error simulator menu you can choose from six different error generating test cases. Selecting any one of the test cases sends a specific message which is interpreted in function MainWndProc in source file generic.f90. For each message, MainWndProc effectively calls the message processing routine ERROR_SIMULATOR to generate the selected condition. But MainWndProc does not call ERROR_SIMULATOR directly. Instead, it calls a wrapper routine, ERROR_SIMULATOR_WRAPPER in source file wrapper.c, which wraps a try-except construct around the call to ERROR_SIMULATOR. Also notice that MainWndProc expects to get a status value returned from ERROR_SIMULATOR_WRAPPER. If ERROR_SIMULATOR_WRAPPER returns a zero, all is well. If a non-zero status is returned, MainWndProc displays a message box indicating the error simulator had an error. The important point to notice is that errors in the simulator routine are contained and turned into status codes that can be acted upon rather than allowing the program to abort when something unexpected occurs. To achieve this behavior, the code must do two things: 1. The simulator must always ask for and test the completion status on Fortran I/O statements. If the I/O status is not FOR$IOS_SUCCESS, the simulator takes appropriate action and returns the non-zero status to MainWndProc. 2. Exceptions must be caught and dealt with. The __except construct in ERROR_SIMULATOR_WRAPPER allows this to happen. When an exception occurs in ERROR_SIMULATOR, the operating system will evaluate the __except filter expression to see if we want to handle the exception. The filter expression is a call to CHECK_EXCEPTION_INFO in generic1.f90. A pointer to the operating system supplied exception information is passed to CHECK_EXCEPTION_INFO. If CHECK_EXCEPTION_INFO returns EXCEPTION_EXECUTE_HANDLER, the code in the __except handler block is executed. The handler block sets the return status to a non-zero value. Execution then continues with the return statement following the handler block and returns to MainWndProc. CHECK_EXCEPTION_INFO takes whatever error handling action is appropriate for the situation. In this sample program, it will clear the floating point exception status for floating point exceptions and in all cases, it will generate a traceback output with a message specific to the exception which occurred. If CHECK_EXCEPTION_INFO returns EXCEPTION_CONTINUE_SEARCH, the operating system looks for any other handlers on its handler list. In this case it
Page 21 of 24
would find the C run-time system default __except filter which calls the C runtime routine _XcptFilter. _XcptFilter would return EXCEPTION_EXECUTE_HANDLER, which would call the C run-time _exit routine to abort the process. Finally, recall that the Fortran run-time system does not supply a C main routine for a Fortran Windows application as in a Fortran Console or Fortran QuickWin application. In those cases, main would have called MAIN__, which would have executed a few compiler generated Fortran run-time system calls to do some additional initialization work. This application compiles with the /fpe:0 option so that floating-point traps can be demonstrated. The application must unmask these traps since the Fortran run-time system will not be called to do it automatically. For this sample, the traps are unmasked at the beginning of ERROR_SIMULATOR using GETCONTROLFPQQ and SETCONTROLFPQQ. The traps would have been masked by default. In some situations, you may want a separate initialization routine to unmask traps for the entire duration of your application. In other cases, you may want to mask or unmask traps at selective points during program execution. If you selectively mask and unmask traps, keep in mind that your code may set floating point exception status bits even through you may have traps masked. The floating-point status bits are "sticky," so before you unmask traps again, you should clear the floating point status bits so you do not generate a false trap later in your code.
C-Style Signal Handling Overview Signal is Really SEH Again How GETEXCEPTIONPTRSQQ Works
C-Style Signal Handling Overview Many Fortran applications were developed on U*X systems where C-style signal handling was the usual way of dealing with exceptions. When ported to Windows, these applications can continue to use the C signal interface. SIGNALQQ will work with any application type using pure Fortran or mixed Fortran and C code. It does not work well, for example, in a VB-VF application.
Page 22 of 24
SIGNALQQ is just a Fortran jacket to the C run-time signal() function. When you call SIGNALQQ, you are actually registering your signal handler (or action) for a particular signal with the C run-time system. The C run-time system simply stores your handler (or action) in an internal exception action table or variable where it associates your handler with the desired signal. The operating system has no knowledge of this association. If you have Visual C++ available, you can look at the code for the C run-time signal routine in ...\MICROSOFT VISUAL STUDIO\VC98\CRT\WINSIG.C and see how the table is managed. The table itself is defined and initialized in source file WINXFLTR.C, available in the same folder. When a signal occurs, the C run-time system checks its internal table to see if you have registered a handler for the particular signal. It calls your routine if you have assigned a handler. Signal is Really SEH Again Notice that it is the C run-time system that calls your handler when a signal occurs, not the operating system. So how did the C run-time get the exception delivered to it? Recall that the entry point of your image is either mainCRTStartup or WinMainCRTStartup, depending on the application type. Refer back to the section Structure of a Visual Fortran Application and look at these entry points (or look at source file Crt0.c in the C run-time sources). Notice that they wrap a try-except construct around a call to either main() or WinMain() and that the filter expression associated with the __except construct calls a function _XcptFilter. _XcptFilter is passed two arguments which are the operating system supplied exception information. When an exception occurs, the operating system looks at the list of exception filters and, starting with the inner most nested try-except construct, evaluates except filter expressions until it finds one which does not return EXCEPTION_CONTINUE_SEARCH. If your application type includes main from the Fortran run-time system and thus the except construct associated with main, the Fortran run-time filter will be evaluated before the C run-time filter. The Fortran filter expression will check to see if you have established your own handler with SIGNALQQ. If it finds there is such a handler, or if you have set the environment variable FOR_IGNORE_EXCEPTIONS, it will return EXCEPTION_CONTINUE_SEARCH to allow the C run-time exception filter the opportunity to deal with the exception and find your handler. If you have not established your own handler or set the environment variable, the Fortran runtime will perform its default exception handling processing. The C filter function, _XcptFilter, compares the exception code from the operating system with its mapping of operating system exceptions to C signal codes. If it finds a match in the table, it uses the exception action entry in the table corresponding to the signal code. This is the same table where your SIGNALQQ handler is recorded as the action for the requested signal code. If
Page 23 of 24
you have established a handler, it will be called from _XcptFilter. Before your handler is called, _XcptFilter resets the specified action for the signal to SIG_DFL in the exception action table. If you try to continue from the exception and you want your handler invoked on the next occurrence of the signal, you must call SIGNALQQ again to reestablish your handler as the action for that signal. When your handler routine is finished executing and returns to _XcptFilter, the value EXCEPTION_CONTINUE_EXECUTION is returned to the operating system by _XcptFilter. The operating system will then resume execution at the point of the exception. If you do not want to continue execution, your handler should take appropriate action to shut down the application. Not every operating system exception code maps to a C signal code. You can see the mapping in source WINXFLTR.C if you have it. Here is the list if you do not have WINXFLTR.C (as of Visual C++ version 6.0): Operating System Exception Code STATUS_ACCESS_VIOLATION STATUS_ILLEGAL_INSTRUCTION STATUS_PRIVILEGED_INSTRUCTION STATUS_FLOAT_DENORMAL_OPERAND STATUS_FLOAT_DIVIDE_BY_ZERO STATUS_FLOAT_INEXACT_RESULT STATUS_FLOAT_INVALID_OPERATION STATUS_FLOAT_OVERFLOW STATUS_FLOAT_STACK_CHECK STATUS_FLOAT_UNDERFLOW How GETEXCEPTIONPTRSQQ Works When the C run-time exception filter function _XcptFilter calls your handler that you established with SIGNALQQ, the only argument passed to your handler is the C signal number. The C run-time system also saves a pointer to the exception information supplied by the operating system. This pointer is named _pxcptinfoptrs and you can retrieve it through the Fortran run-time routine GETEXCEPTIONPTRSQQ. See C header file signal.h for the public definition of _pxcptinfoptrs. C Signal Number SIGSEGV SIGILL SIGILL SIGFPE SIGFPE SIGFPE SIGFPE SIGFPE SIGFPE SIGFPE
Page 24 of 24
The value returned by GETEXCEPTIONPTRSQQ can be used in your handler routine to generate a traceback with TRACEBACKQQ. GETEXCEPTIONPTRSQQ just returns _pxcptinfoptrs. This pointer is only valid while you are executing within the evaluation of the C run-time filter function _XcptFilter because the exception information is on the program stack, so do not use GETEXCEPTIONPTRSQQ in any other context. See the sample program ...\SAMPLES\EXCEPTIONHANDLING\GETEPTRS for a working example of using GETEXCEPTIONPTRSQQ.
Page 1 of 10
Supported Native and Nonnative Numeric Formats Limitations of Numeric Conversion Methods of Specifying the Data Format Environment Variable FORT_CONVERTn Method Environment Variable FORT_CONVERT.xxx or FORT_CONVERT_xxx Method OPEN Statement CONVERT='keyword' Method OPTIONS Statement Method Compiler Option /convert:keyword Method Additional Notes on Nonnative Data
If your program needs to read or write unformatted data files containing a floating-point format that differs from the format in memory for that data size, you can request that the unformatted data be converted. Data storage in different computers uses a convention of either little endian or big endian storage. The storage convention generally applies to numeric values that span multiple bytes, as follows:
l
Little endian storage occurs when: The least significant bit (LSB) value is in the byte with the lowest address. The most significant bit (MSB) value is in the byte with the highest address. The address of the numeric value is the byte containing the LSB.
Page 2 of 10
Big endian storage occurs when: The least significant bit (LSB) value is in the byte with the highest address. The most significant bit (MSB) value is in the byte with the lowest address. The address of the numeric value is the byte containing the MSB. Subsequent bytes with higher addresses contain less significant bits.
The following figure shows the difference between the two byte-ordering schemes. Little and Big Endian Storage of an INTEGER Value
Moving unformatted data files between big endian and little endian computers requires that the data be converted. Visual Fortran provides the capability for programs to read and write unformatted data (originally written using unformatted I/O statements) in several nonnative floating-point formats and in big endian INTEGER or floatingpoint format. Supported nonnative floating-point formats include Compaq VAX little endian floating-point formats supported by VAX FORTRAN, standard IEEE big endian floating-point format found on most Sun MicrosystemsTM systems and IBM RISC System/6000 systems, IBM floating-point formats (associated with the IBM's System/370 and similar systems), and CRAY floating-point formats. Converting unformatted data instead of formatted data is generally faster and is less likely to lose precision of floating-point numbers. The native memory format includes little endian integers and little endian IEEE floating-point formats, S_float for REAL(KIND=4) and COMPLEX(KIND=4) declarations and T_float for REAL(KIND=8) and COMPLEX(KIND=8) declarations. The keywords for supported nonnative unformatted file formats and their data
Converting Unformatted Numeric Data types are listed in the following table:
Page 3 of 10
Nonnative Numeric Formats, Keywords, and Supported Data Types Keyword BIG_ENDIAN Description Big endian integer data of the appropriate size (one, two, or four bytes) and big endian IEEE floating-point (REAL (KIND=4), REAL(KIND=8), COMPLEX(KIND=4), COMPLEX (KIND=8)) formats of the appropriate size for either real or complex numbers. INTEGER(KIND=1) data is the same for little endian and big endian. Big endian integer data of the appropriate size (one, two, four, or eight bytes) and big endian CRAY proprietary floating-point format of size REAL(KIND=8) or COMPLEX (KIND=8). Little endian integer data of the appropriate size (one, two, four, or eight bytes) and Compaq VAX floating-point data of format F_floating for REAL(KIND=4) or COMPLEX(KIND=4), and D_Floating for REAL(KIND=8) or COMPLEX(KIND=8). Little endian integer data of the appropriate size (one, two, four, or eight bytes) and Compaq VAX floating-point data of format F_floating for REAL(KIND=4) or COMPLEX(KIND=4), and G_Floating for REAL(KIND=8) or COMPLEX(KIND=8). Big endian integer data of the appropriate size (one, two, or four bytes) and big endian IBM proprietary floating-point format of size REAL(KIND=4) or COMPLEX(KIND=4) or size REAL(KIND=8) or COMPLEX(KIND=8). Native little endian integers of the appropriate size (one, two, four, or eight bytes) and native little endian IEEE floating-point data of the appropriate size and type (REAL (KIND=4), REAL(KIND=8), COMPLEX(KIND=4), COMPLEX (KIND=8)). These are the same formats as stored in memory. For additional information on supported ranges for these data types, see Native IEEE Floating-Point Representations. No conversion occurs between memory and disk. This is the default for unformatted files.
CRAY
FDX
FGX
IBM
LITTLE_ENDIAN
NATIVE
Page 4 of 10
Little endian integers of the appropriate size (one, two, four, or eight bytes) and Compaq VAX floating-point format F_floating for size REAL(KIND=4) or COMPLEX(KIND=4), and D_floating for size REAL(KIND=8) or COMPLEX (KIND=8). Little endian integers of the appropriate size (one, two, four, or eight bytes) and Compaq VAX floating-point format F_floating for size REAL(KIND=4) or COMPLEX(KIND=4), and G_floating for size REAL(KIND=8) or COMPLEX (KIND=8).
VAXG
When reading a nonnative format, the nonnative format on disk is converted to native format in memory. If a converted nonnative value is outside the range of the native data type, a run-time message is displayed.
Binary data (FORM='BINARY') Formatted data (FORM='FORMATTED') Unformatted data (FORM='UNFORMATTED') written by Microsoft Fortran PowerStation or by Visual Fortran with the /fpscomp:ioformat compiler option in effect.
Page 5 of 10
unformatted data. If none of these methods are specified, the native LITTLE_ENDIAN format is assumed (no conversion occurs between disk and memory). Any keyword listed in Supported Native and Nonnative Numeric Formats can be used with any of these methods. The five methods you can use to specify the type of nonnative (or native) format are as follows: 1. Setting an environment variable for a specific unit number before the file is opened. The environment variable is named FORT_CONVERTn, where n is the unit number. 2. Setting an environment variable for a specific file name extension before the file is opened. The environment variable is named FORT_CONVERT.ext or FORT_CONVERT_ext, where ext is the file name extension (suffix). 3. Compiling the program with an OPTIONS statement that specifies the /CONVERT=keyword qualifier. This method affects all unit numbers using unformatted data specified by the program. 4. Specifying the CONVERT keyword in the OPEN statement for a specific unit number. 5. Compiling the program with the appropriate compiler option (DF command /convert: keyword or visual development environment equivalent), which affects all unit numbers that use unformatted data specified by the program. If you specify more than one method, the order of precedence when you open a file with unformatted data is to: 1. Check for an environment variable (FORT_CONVERTn) for the specified unit number (applies to any file opened on a particular unit). 2. Check for an environment variable (FORT_CONVERT.ext is checked before FORT_CONVERT_ext) for the specified file name extension (applies to all files opened with the specified file name extension). 3. Check the OPEN statement CONVERT specifier. 4. Check whether an OPTIONS statement with a /CONVERT=keyword qualifier was present when the program was compiled. 5. Check whether the compiler option /convert:keyword was present when the program was compiled. The following sections describe each method:
l l l l l
Environment Variable FORT_CONVERTn Method Environment Variable FORT_CONVERT.xxx or FORT_CONVERT_xxx Method OPEN Statement CONVERT= Method OPTIONS Statement Method Compiler Option /convert Method
Page 6 of 10
The following figure shows the data formats used on disk and in memory when the example file c:\users\leslie\convieee.exe is run after the environment variables are set. Sample Unformatted File Conversion
Page 7 of 10
The FORT_CONVERTn method takes precedence over this method. When the appropriate environment variable is set when you open the file, the FORT_CONVERT.ext or FORT_CONVERT_ext environment variable is used if a FORT_CONVERTn environment variable is not set for the unit number. The FORT_CONVERTn and the FORT_CONVERT.ext or FORT_CONVERT_ext environment variable methods take precedence over the other methods. For instance, you might use this method to specify that a unit number will use a particular format instead of the format specified in the program (perhaps for a one-time file conversion). You can set the appropriate environment variable using the format FORT_CONVERT.ext or FORT_CONVERT_ext. If you also use Compaq Fortran on U*X systems, consider using the FORT_CONVERT_ext form, because a dot (.) cannot be used for environment variable names on certain U*X command shells.
Page 8 of 10
If you do define both FORT_CONVERT.ext and FORT_CONVERT_ext for the same extension (ext), the file defined by FORT_CONVERT.ext is used. On Windows systems, the file name extension (suffix) is not case-sensitive. The extension must be part of the file name (not the directory path).
A hard-coded OPEN statement CONVERT keyword value cannot be changed after compile time. However, to allow selection of a particular format at run time, equate the CONVERT keyword to a variable and provide the user with a menu that allows selection of the appropriate format (menu choice sets the variable) before the OPEN occurs. You can also select a particular format at run time for a unit number by using one of the environment variable methods (FORT_CONVERTn, FORT_CONVERT.ext, or FORT_CONVERT_ext), which take precedence over the OPEN statement CONVERT keyword method.
Page 9 of 10
The FORT_CONVERTn environment variable method, FORT_CONVERT.ext or FORT_CONVERT_ext environment variable method, and the OPEN statement CONVERT keyword method take precedence over this method. For instance, you might use the FORT_CONVERTn environment variable or OPEN CONVERT keyword method to specify each unit number that will use a format other than that specified using the DF option method. This method takes precedence over the DF /convert:keyword compiler option method. You can use OPTIONS statements to specify the appropriate floating-point formats (in memory and in unformatted files) instead of using the corresponding DF command qualifiers. For example, to use VAX F_floating and G_floating as the unformatted file format, specify the following OPTIONS statement:
OPTIONS /CONVERT=VAXG
Because this method affects all unit numbers, you cannot read data in one format and write it in another format, unless you use it in combination with one of the environment variable methods or the OPEN statement CONVERT keyword method to specify a different format for a particular unit number. For more information, see the OPTIONS statement.
Page 10 of 10
Because this method affects all unformatted file unit numbers, you cannot read data in one format and write it in another file format using the /convert:keyword compiler option method alone. You can if you use it in combination with the environment variable methods or the OPEN statement CONVERT keyword method to specify a different format for a particular unit number.
When porting source code along with the unformatted data, vendors might use different units for specifying the record length (RECL specifier) of unformatted files. While formatted files are specified in units of characters (bytes), unformatted files are specified in longword units for Compaq Fortran (default) and some other vendors. To allow you to specify the RECL units (bytes or longwords) for unformatted files without source file modification, use the /assume:byterecl compiler option (in the visual development environment, this is available in the Project menu Settings item, Fortran tab, Fortran Data category). The Fortran 90 standard (American National Standard Fortran 90, ANSI X3.198-1991, and International Standards Organization standard ISO/IEC 1539:1991), in Section 9.3.4.5, states: "If the file is being connected for unformatted input/output, the length is measured in processor-dependent units."
Certain vendors apply different OPEN statement defaults to determine the record type. The default record type (RECORDTYPE) with Compaq Fortran depends on the values for the ACCESS and FORM specifiers for the OPEN statement (also described in the Compaq Fortran Language Reference Manual). Certain vendors use a different identifier for the logical data types, such as hex FF instead of 01 to denote "true." Source code being ported may be coded specifically for big endian use.
Hexadecimal-Binary-Octal-Decimal Conversions
Page 1 of 1
Hexadecimal-Binary-Octal-Decimal Conversions
The following table lists hexadecimal, binary, octal, and decimal conversion: Hexadecimal, Binary, Octal, and Decimal Conversion Hex Number 0 1 2 3 4 5 6 7 8 9 A B C D E F Binary Number 0000 0001 0010 0011 0100 0101 0110 0111 1000 1001 1010 1011 1100 1101 1110 1111 Octal Number 00 01 02 03 04 05 06 07 10 11 12 13 14 15 16 17 Decimal Number 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
Page 1 of 6
README.TXT
This ASCII readable file contains the Release Notes for the IMSL Libraries. It also includes information on accessing the IMSL libraries.
l
On-line PDF documentation The IMSL online Adobe Acrobat PDF documentation lets you quickly find details on the purpose and use of any IMSL Library routine. You can access the following topics through the online IMSL Routines Reference (PDF version):
Mathematical Library Subroutines Mathematical Library Special Functions Statistical Library Subroutines, Volumes 1 and 2 Fortran 90 MP Library Subroutines
Click on any of the libraries to see a submenu of items grouped by subject. Within each category (for example, Linear Systems, Eigensystem Analysis, and so on), click on a routine for more information. IMSL libraries (and IMSL online documentation files) are included with the Professional and Enterprise Editions of Visual Fortran (not the Standard Edition). When calling the IMSL library routines from a multi-threaded application, be aware that the IMSL routines are intended only for single-thread use. This section provides information on the following topics:
l l
Using the IMSL Libraries from Visual Fortran Library Naming Conventions
Page 2 of 6
d. Click OK if you have changed any information. 3. Make IMSL routines and their interfaces available to your program:
When calling Mathematics and Statistical library routines from a Fortran program, you should use the numerical_libraries module to provide interface blocks and parameter definitions for the routines.
Page 3 of 6
Including the following USE statement in your calling program will verify the correct usage of the IMSL routines at compile time:
USE numerical_libraries
When you add the USE numerical_libraries line to your program, you should not declare the called routines as external with the EXTERNAL statement. For more details, see the IMSL readme file in the Visual Fortran program folder. When calling Mathematics and Statistical library routines, you do not need to declare the functions or subroutines separately.
When also calling Fortran 90 MP library routines, you should instead use the imslf90 module to provide interface blocks and parameter definitions for all the Fortran 90 MP routines and the MATH and STAT library routines. Including the following USE statement in your calling program will verify the correct usage of the IMSL routines at compile time:
USE IMSLF90
For more information about calling the Fortran 90 MP routines, see the IMSL Libraries online PDF file. The free-form Fortran 95/90 example program below invokes the function AMACH and the subroutine UMACH from the IMSL Libraries. The AMACH function retrieves real machine constants that define the computer's real arithmetic. A value for positive machine infinity is returned (Infinity). The subprogram UMACH retrieves the output unit number.
! This free-form example demonstrates how to call ! IMSL routines from Visual Fortran. ! ! The module numerical_libraries includes the Math and ! Stat libraries; these contain the type declarations ! and interface statements for the library routines. PROGRAM SHOWIMSL USE NUMERICAL_LIBRARIES INTEGER NOUT REAL RINFP ! The AMACH function and UMACH subroutine are ! declared in the numerical_libraries module CALL UMACH(2,NOUT) RINFP = AMACH(7)
Page 4 of 6
For information on compiling and linking with the visual development environment, see Building Programs and Libraries. Note: IMSL routines are in general not multithread safe. In a multithread environment, you should take care that no two IMSL routines are active at the same time. To insure this, use multithread control techniques. For further information, see Creating Multithread Applications.
Library Description IMSL Static Library, contains FORTRAN 77 and Fortran 90 routines IMSL Error Handler Library Stub Library for MPI Routines used in IMSL
IMSLS_ERR IMSLMPISTUB
The IMSL Numerical Libraries are for applications in general applied mathematics and for analyzing and presenting statistical data in scientific and business applications. For command-line window development, executing the DFVARS.BAT file (see Using the Compiler and Linker from the Command Line) sets environment variables required by Visual Fortran. For more information on the IMSL libraries, see:
l l
The IMSL readme file provided in the Visual Fortran program folder. The IMSL online documentation PDF files provided in the Compaq Visual Fortran program folder for the IMSL Routines Reference. Product information about IMSL at the following Internet URL: http://www.vni.com
Page 5 of 6
Messages that IMSL routines write to standard output or to error output in a mixed-language application or an application for Windows can be awkward if they are written to the screen. You can avoid this by calling UMACH from a Fortran routine to remap the output and error units to a file instead of to the screen. For example, the following free-form program writes the standard output from VHSTP to the file STD.TXT, and the error message from AMACH to the file ERR.TXT:
! ! ! ! ! PROGRAM fileout This program demonstrates how to use the UMACH routine to redirect the standard output and error output from IMSL routines to files instead of to the screen. The routines AMACH and UMACH are declared in the numerical_libraries module USE numerical_libraries INTEGER STDU, ERRU REAL x, frq(10)/3.0,1.0,4.0,1.0,5.0,9.0,2.0,6.0,5.0,3.0/ ! ! ! Redirect IMSL standard output to STD.TXT at unit 8 CALL umach(-2, STDU) OPEN (unit=STDU, file='std.txt') CALL vhstp(10,frq,1,'Histogram Plot') CLOSE(8) ! ! ! Redirect IMSL error output to ERR.TXT at unit 9 CALL umach(-3, ERRU) OPEN (unit=ERRU, file='err.txt') x = amach(0) ! Illegal parameter error CLOSE(9) END
The standard output from IMSL routine VHSTP written to STD.TXT is:
1 Histogram Plot Frequency------------------------9 * I * 8 * I * 7 * I * 6 * I I * 5 * I I I I * 4 * I I I I I * 3 * I I I I I I I * 2 * I I I I I I I I * 1 * I I I I I I I I I I * ---------------------------------Class 5 10
The error output from IMSL routine AMACH written to ERR.TXT is:
*** TERMINAL ERROR 5 from AMACH. The argument must be between 1 and 8 *** inclusive. N = 0
Consider the following simple Fortran example that uses the IMSL library:
Page 6 of 6
USE numerical_libraries real rinfp rinfp = AMACH(7) write(*,*) 'Real positive machine infinity = ',rinfp end
The _stdcall modifier in the function prototype needed when calling the IMSL libraries. The & address operator passes the address of the variable to the subprogram (IMSL libraries expect arguments passed by reference).
The C example can be compiled by the cl command to create an object file that can be linked using the DF command. For more information on mixed-language programming, see Programming with Mixed Languages.
Page 1 of 3
Page 2 of 3
Name
Basic Linear Algebra
Description The Basic Linear Algebra Subprograms (BLAS) library includes the industry-standard Basic Linear Algebra Subprograms for Level 1 (vector-vector, BLAS1), Level 2 (matrix-vector, BLAS2), and Level 3 (matrix-matrix, BLAS3). Also included are subprograms for BLAS Level 1 Extensions, and Sparse BLAS Level 1. The Signal Processing library provides a basic set of signal processing functions. Included are one-, two-, and threedimensional Fast Fourier Transforms (FFT), group FFTs, Cosine/Sine Transforms (FCT/FST), Convolution, Correlation, and Digital Filters. The Sparse Linear System library provides both direct and iterative sparse linear system solvers. The direct solver package supports both symmetric and nonsymmetric sparse matrices. The iterative solver package contains a basic set of storage schemes, preconditioners, and iterative solvers. LAPACK is an industry-standard subprogram package offering an extensive set of linear system and eigenproblem solvers. LAPACK uses blocked algorithms that are better suited to most modern architectures, particularly ones with memory hierarchies. Utility subprograms include random number generation, vector math functions, and sorting subprograms.
Signal Processing
LAPACK
Utility subprograms
Where appropriate, each subprogram has a version to support each combination of real or complex and single or double precision arithmetic. For command-line window development, executing the DFVARS.BAT file (see Using the Compiler and Linker from the Command Line) sets Visual Fortran environment variables as well as CXML environment variables (see Environment Variables Used with the DF Command). If you have problems during linking related to CXML routines, please see the online release notes. For information on compiling and linking with the visual development environment, see Building Programs and Libraries. For more information on CXML, refer to:
l
Page 3 of 3
CXML readme file provided in the Compaq Visual Fortran program folder. CXML Reference Guide, available in PDF format. CXML Samples, installed in ...\Df98\CXML\Samples. Compaq Visual Fortran Installing and Getting Started guide.
An example of the INCLUDE statement appears in the following example program. CXML Program Example The free-form Fortran example program below invokes the function SAXPY from the BLAS portion of the CXML Libraries. The SAXPY function computes a*x+y.
PROGRAM example ! ! This free-form example demonstrates how to call ! CXML routines from Visual Fortran. ! ! The include file CXML_INCLUDE.F90 contains interface statements for ! the CXML library routines and causes the linker to search the CXML ! library. It must be the first statement following the statement ! that introduces the program unit. ! INCLUDE 'CXML_INCLUDE.F90' REAL(KIND=4) :: a(10) REAL(KIND=4) :: b(10) REAL(KIND=4) :: alpha INTEGER(KIND=4) :: n INTEGER(KIND=4) :: incx INTEGER(KIND=4) :: incy n = 5 ; incx = 1 ; incy = 1 ; alpha = 3.0 DO i = 1,n a(i) = FLOAT(i) b(i) = FLOAT(2*i) ENDDO PRINT 98, (a(i),i=1,n) PRINT 98, (b(i),i=1,n) 98 FORMAT(' Input = ',10F7.3) CALL saxpy( n, alpha, a, incx, b, incy ) PRINT 99, (b(i),I=1,n) 99 FORMAT(/,' Result = ',10F7.3) STOP END PROGRAM example
Compatibility Information
Page 1 of 16
Compatibility Information
Visual Fortran uses the same Compaq Fortran compiler available on Tru64 UNIX (formerly DIGITAL UNIX) Alpha, Linux Alpha, and OpenVMS Alpha systems. Compaq Visual Fortran supports extensions to the ISO and ANSI standards, including a number of extensions defined by:
l l
Compaq Fortran for the various Compaq Fortran Alpha platforms Microsoft Fortran PowerStation 4.0
Many language extensions associated with Microsoft Fortran PowerStation Version 4 have been added to Visual Fortran; most of these extensions have been added to different releases of Compaq Fortran on Alpha platforms. The following sections describe Visual Fortran compatibility information:
l l
Compatibility with Compaq Fortran on Other Platforms Compatibility with Microsoft Fortran PowerStation
Compaq Fortran (formerly Digital Fortran) Compaq Fortran (including Compaq Visual Fortran) provides many of the language extensions provided by Compaq Fortran 77 for OpenVMS VAX Systems (formerly DEC Fortran 77 and VAX FORTRAN).
Microsoft Fortran PowerStation 4.0 For information about Microsoft Fortran PowerStation 4.0, see Compatibility with Microsoft Fortran PowerStation.
Summary of Language Compatibility Common Language Extensions Major Changes from the FORTRAN 77 to Fortran 95 Standards Platform Porting Notes
Compatibility Information
Page 2 of 16
Compaq Visual Fortran (CVF) uses the same robust, highly efficient Fortran 95/90 compiler used by Compaq Fortran for the following Alpha operating systems:
l l l
Compaq Tru64 UNIX Alpha systems Linux Alpha systems Compaq OpenVMS Alpha systems
The following table summarizes the compatibility of Compaq Visual Fortran with Compaq Fortran for Alpha systems and Compaq Fortran 77 (CF77) for OpenVMS VAX systems: Summary of Language Compatibility Compaq Fortran (CF) OS/Architecture Platform Language Feature Linking against static and shared libraries Create code for shared libraries Recursive code support AUTOMATIC and STATIC statements STRUCTURE and RECORD declarations INTEGER*1, *2, *4 LOGICAL*1, *2, *4 INTEGER*8 and LOGICAL*8 REAL*4, *8 REAL*161 COMPLEX*8, *16 COMPLEX*322 X CF CVF UNIX Windows Alpha X X X X X X X X X X X X X X X X X X X X X CF Linux Alpha X X X X X X X X X X X X CF OpenVMS Alpha X X X X X X X X X X X X X X X CF77 OpenVMS VAX X X X X X X X
Compatibility Information POINTER (CRAY-style) INCLUDE statement IMPLICIT NONE statement Data initialization in type declarations Automatic arrays VOLATILE statements NAMELIST-directed I/O 31-character names including $ and _ Source listing with machine code Debug statements in source Bit constants to initialize data and use in arithmetic DO WHILE and END DO statements Built-in functions %LOC, %REF, %VAL SELECT CASE construct EXIT and CYCLE statements Variable FORMAT expressions (VFEs) ! marks end-of-line comment Optional run-time bounds checking for arrays and substrings X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X
Page 3 of 16 X X X X
X X X X X
X X X X X X
X X X X X X
X X X X X X
X X X X X X
X X
X X
Compatibility Information Binary (unformatted) I/O in IEEE big endian, IEEE little endian, VAX, IBM, and CRAY floatingpoint formats Fortran 95/90 standards checking FORTRAN-77 standards checking IEEE exception handling VAX floating data type in memory IEEE floating data type in memory CDD/Repository DICTIONARY support KEYED access and INDEXED files Parallel decomposition OpenMP parallel directives Conditional compilation using IF...DEF constructs Vector code support Direct inlining of Basic Linear Algebra Subroutines (BLAS) DATE_AND_TIME returns 4-digit year FORALL statement and construct Automatic deallocation of ALLOCATABLE arrays Dim argument to MAXLOC and MINLOC
5 5 5 5 4
Page 4 of 16
X X
X X X
X X X X X
X3,4 X
X X X
X X X X
X X X X
X X X X
X X X X
Compatibility Information PURE user-defined subprograms ELEMENTAL user-defined subprograms Pointer initialization (initial value) The NULL intrinsic to nullify a pointer Derived-type structure initialization CPU_TIME intrinsic subroutine Kind argument to CEILING and FLOOR intrinsics Nested WHERE constructs, masked ELSEWHERE statement, and named WHERE constructs Comments allowed in namelist input Generic identifier in END INTERFACE statements Minimal FORMAT edit descriptor field width Detection of Obsolescent and/or Deleted features6
Page 5 of 16
X X X X X X
X X X X X X
X X X X X X
X X X X X X
X X X X
X X X X
X X X X
X X X X
For REAL*16 data, OpenVMS VAX systems use H_float format, and Alpha systems use IEEE style X_float format. For COMPLEX*32 data, Alpha systems use IEEE style X_float format for both REAL*16 parts.
Compatibility Information
3
Page 6 of 16
For parallel processing with Compaq Fortran for Compaq Tru64 UNIX Systems, you can use the OpenMP or Compaq Fortran directives on shared memory multiprocessor systems or MPI software for using High Performance Fortran (HPF) constructs across multiple systems. For parallel processing, you can use the optional KAP performance preprocessor for a shared memory multiprocessor system. BLAS and other routines are available with the Compaq Extended Mathematical Library (CXML) software provided with Compaq Visual Fortran and Compaq Fortran for Alpha systems (for download information, see http://www.compaq.com/math). Compaq Fortran flags these deleted and obsolescent features, but fully supports them.
4 5
l l
l l
l l l l
l l l l l l l
Record structures (STRUCTURE and RECORD statements) I/O statements, including PRINT, ACCEPT, TYPE, DELETE, and UNLOCK I/O statement specifiers, such as the INQUIRE statement specifiers CARRIAGECONTROL, CONVERT, ORGANIZATION, and RECORDTYPE Multiple INTEGER and REAL kinds Size specifiers for data declaration statements, such as INTEGER*4, in addition to the KIND type parameter IEEE floating-point data types in memory The POINTER statement and its associated data type (integer pointers or CRAY pointers). The typeless PARAMETER statement The VOLATILE statement The AUTOMATIC and STATIC statements Built-in functions used in argument lists, such as %REF, %VAL, and % LOC Hollerith constants Variable-format expressions (VFEs) Certain intrinsic functions The tab source form (a variation of fixed-source form) I/O formatting descriptors USEROPEN routines for user-defined open routines Additional language features, including the DEFINE FILE, ENCODE, DECODE, and VIRTUAL statements
Compatibility Information
Page 7 of 16
In addition to language extensions, Compaq Visual Fortran also supports the following Compaq Fortran 77 features:
l
Compaq Fortran 77 compilation control statements and directives, including: INCLUDE statement forms using /LIST and /NOLIST (requires compiling with the /vms compiler option ) OPTIONS statement to override or set compiler command-line options General cDEC$ directives, including: n cDEC$ ALIAS n cDEC$ IDENT n cDEC$ OPTIONS n cDEC$ PSECT n cDEC$ TITLE n cDEC$ SUBTITLE foriosdef.for symbolic parameter definitions for use with run-time (IOSTAT) error handling. Compaq Visual Fortran includes the following features and enhancements also found on other Compaq Fortran platforms:
Support for linking against static libraries Support for linking against dynamically linked libraries (DLL) Support for creating code to be put into a dynamically linked library (DLL) Support for stack-based storage Support for dynamic memory allocation Support for reading and writing binary data files in nonnative formats, including IEEE (little-endian and big-endian), VAX, IBM System\360, and CRAY integer and floating point formats User control over IEEE floating point exception handling, reporting, and resulting values Control for memory boundary alignment of items in COMMON and fields in structures and warnings for misaligned data Directives to control listing page titles and subtitles, object file identification field, COMMON and record field alignment, and some attributes of COMMON blocks Ability to CALL an external function subprogram 7200 character statement length Free form unlimited line length Composite data declarations using STRUCTURE, END STRUCTURE, and RECORD statements, and access to record components through field references Explicit specification of storage allocation units for data types such as: n INTEGER*4 n LOGICAL*4
Compatibility Information
n
Page 8 of 16
REAL*4 n REAL*8 n COMPLEX*8 Support for 64-bit signed integers using INTEGER*8 and LOGICAL*8 Support for 128-bit floating-point data using REAL*16 and COMPLEX*32 (on Alpha platforms only) A set of data types: n BYTE n LOGICAL*1, LOGICAL*2, LOGICAL*4, LOGICAL*8 n INTEGER*1, INTEGER*2, INTEGER*4, INTEGER*8 n REAL*4, REAL*8 n COMPLEX*8, COMPLEX*16, DOUBLE COMPLEX Compaq Fortran POINTER statement (CRAY style) Data statement style initialization in type declaration statements AUTOMATIC and STATIC statements Bit constants to initialize LOGICAL, REAL, and INTEGER values and participate in arithmetic and logical expressions Built-in functions %LOC, %REF, and %VAL VOLATILE statement Bit manipulation functions Binary, hexadecimal, and octal constants and Z and O format edit descriptors applicable to all data types I/O unit numbers that can be any nonnegative INTEGER*4 value Variable amounts of data can be read from and written to "STREAM" files, which contain no record delimiters ENCODE and DECODE statements ACCEPT, TYPE, and REWRITE input/output statements DEFINE FILE, UNLOCK, and DELETE statements USEROPEN subroutine invocation at file OPEN Support for reading nondelimited character strings as input for character NAMELIST items Debug statements in source Generation of a source listing file with optional machine code representation of the executable source Variable format expressions in a FORMAT statement Optional run-time bounds checking of array subscripts and character substrings 31-character identifiers that can include dollar sign ($) and underscore (_) Support for the supercomputer intrinsics POPCNT, POPPAR, LEADZ, and TRAILZ Language elements that support the various extended range and extended precision floating point architectural features: n 32-bit IEEE S_floating data type, with an 8-bit exponent and 24bit mantissa and a precision of typically 7 decimal digits n 64-bit IEEE T_floating data type, with an 11-bit exponent and 53bit mantissa and a precision of typically 15 decimal digits
Compatibility Information
Page 9 of 16
Command line control for: n The size of default INTEGER, REAL, and DOUBLE PRECISION data items n The levels and types of optimization to be applied to the program n The directories to search for INCLUDE and module files n Inclusion or suppression of various compile-time warnings n Inclusion or suppression of run-time checking for various I/O and computational errors n Control over whether compilation terminates after a specific number of errors has been found n Choosing whether executing code will be thread-reentrant n Internal procedures can be passed as actual arguments to procedures Kind types for all of the hardware-supported data types: n For 1-, 2-, 4-, and 8-byte LOGICAL data: LOGICAL (KIND=1), LOGICAL (KIND=2), LOGICAL (KIND=4), LOGICAL (KIND=8) n For 1-, 2-, 4-, and 8-byte INTEGER data: INTEGER (KIND=1), INTEGER (KIND=2), INTEGER (KIND=4), INTEGER (KIND=8) n For 4- and 8-byte REAL data: REAL (KIND=4), REAL (KIND=8) n For single precision and double precision COMPLEX data: COMPLEX (KIND=4), COMPLEX (KIND=8)
Array operations and features Modules Free-form source Improved facilities for numeric computation Parameterized intrinsic data types User-defined (derived) data types and operators Generic user-defined procedures Interface blocks Pointers (Fortran 90 pointers) The concept of language evolution Additional features for source text Optional procedure arguments Additional input/output features Additional control constructs Additional intrinsic procedures Additional specification statements Additional way to specify attributes Scope and Association
Page 10 of 16
Support for recursive subprograms IMPLICIT NONE statements INCLUDE statement NAMELIST-directed I/O DO WHILE and ENDDO statements Use of exclamation point (!) for end of line comments Support for automatic arrays Support for the following SELECT CASE CASE CASE DEFAULT END SELECT statements Support for the EXIT and CYCLE statements and for construct names on DO END DO statements
Major changes and improvements to the Fortran 90 standard introduced by the Fortran 95 standard include:
l l l l l l l l l
l l l l l
The FORALL statement and construct PURE user-defined procedures ELEMENTAL user-defined procedures CPU_TIME intrinsic subroutine NULL intrinsic function Derived-type structure default initialization Pointer initialization Automatic deallocation of allocatable arrays Enhanced intrinsic functions, including CEILING, FLOOR, MAXLOC, MINLOC, and SIGN Printing of -0.0 Enhanced WHERE construct Generic identifier allowed in END INTERFACE statement Minimum-length formats Comments allowed in namelist input
The use of the /stand option during compilation can help identify source code that uses language extensions during compilation. On U*X Alpha systems, specify the -std option. On OpenVMS systems, specify the /STANDARD qualifier. The online Visual Fortran Language Reference and the printed Compaq Fortran Language Reference Manual identify
Compatibility Information
Page 11 of 16
language extensions with a green-blue color. The use of certain platform-specific procedures (such as library routines) provided by the operating system need to be located and likely changed. Whenever possible, use Fortran standard language calls (such as to intrinsic procedures) instead of platform-specific routines. For example, if you attempt to link source files and the linker returns unresolved references/symbols, be aware that the routine names may be platformspecific: When porting code from a U*X system, identify code that uses system calls (section 2 in reference pages) and certain other library routines not found on Windows systems. Visual Fortran provides the portability library that includes many U*X Fortran library routines (section 3f in reference pages) found on U*X systems. The Compaq Extended Math Library (CXML) routines supported on Alpha systems are provided with Visual Fortran. For more details about the U*X Alpha programming environment, see the appropriate sections in the Compaq Fortran User Manual for Tru64 UNIX and Linux Systems. When porting an application from an OpenVMS system, identify code that calls system services (SYS$ prefix) or RTL routines (such prefixes as LIB$, CVT$, DTK$, MATH$, OTS$, PPL$, SMG$, and STR$). The Compaq Extended Math Library (CXML) routines supported on Alpha systems are provided with Visual Fortran. Character arguments on OpenVMS systems may be passed by character descriptors not found on other platforms. For more details about the OpenVMS Alpha programming environment, see the appropriate sections in the Compaq Fortran User Manual for OpenVMS Alpha Systems.
When porting an application from Visual Fortran to an Alpha system, non-graphical applications are usually easier to port. The nature of Visual Fortran applications is usually defined by the chosen project type (such as Fortran Console applications). For example, applications that use many platform-specific graphical system calls are usually difficult to port to other platforms. Certain platform-specific Fortran routine names either have QQ appended at the end of the name or use the Win32 naming conventions (see Calling Win32 Routines). Certain groups of routines require different USE statements. You can use the HTML Help viewer's index and search facility to locate descriptions of Visual Fortran or Win32 routines. The online Visual Fortran Language Reference and the printed Compaq Fortran Language Reference Manual identify language elements that are supported only on certain Compaq Fortran platforms.
Compatibility Information
Page 12 of 16
Certain math library routines provided by Visual Fortran are available on other platforms: n The IMSL Math and Statistical routines provided with the Visual Fortran Professional or Enterprise Editions are also available for many other platforms (including Compaq Alpha systems) from Visual Numerics, Inc. (see http://www.vni.com). n The Compaq Extended Math Library (CXML) routines provided with all Visual Fortran Editions are also provided with Compaq Fortran on Alpha systems. CXML routines are available separately for Compaq Alpha systems (see http://www.compaq.com/math).
Certain application code may make assumptions about the numeric data format stored in memory. Compaq Fortran and Visual Fortran store numeric data in memory in little endian format, while some vendors use big endian formats. Also, certain platforms support different sized intrinsic data types. Certain application code may make assumptions about the size of Compaq Fortran (integer) pointers. For example, on an U*X Alpha system, the size of an address is 64 bits. On Windows ia32 systems, the size of an address is 32 bits. On Windows ia64 systems, the size of an address is 64 bits. Whenever possible, use the standard-conforming Fortran 95/90 pointers instead of Compaq Fortran pointers. Certain platforms use different name decoration conventions, such as appending an extra character to symbol names. Use of the appropriate cDEC$ ATTRIBUTES directives can help overcome this problem (see Adjusting Naming Conventions in Mixed-Language Programming. Porting data files used by an application from one platform to another. Consider using ftp for binary file transfers to preserve file integrity. Visual Fortran provides the capability to convert unformatted data files in various formats, including big endian formats (see Supported Native and Nonnative Numeric Formats). It is also important to select compatible cross-platform record types. When porting data files between Windows systems running Compaq Visual Fortran and Alpha systems running Compaq Fortran: 1. For formatted data files, use the stream_LF record type. 2. For unformatted files, use the segmented record type. Unformatted files ported from Compaq Fortran OpenVMS systems may use a nonIEEE floating-point format. For example, Compaq Fortran for OpenVMS Alpha systems allows the user to specify either IEEE and or VAX formats during compilation (see Supported Native and Nonnative Numeric Formats).
Compatibility Information
Page 13 of 16
One suggested way to port an unformatted file from an OpenVMS system follows: 1. From the OpenVMS system type the following DCL command for the file (file.dat):
$ SET FILE /ATTR=(RFM=FIX,LRL=512) file.dat
For information on porting unformatted data files, see the Compaq Fortran Knowledge Base at http://www.compaq.com/fortran/kb. Examples of record type conversion programs are available at http://www.compaq.com/fortran/examples. 2. Copy the file to your Windows system as a binary (image) copy (perhaps using ftp). 3. Modify the source code to OPEN the file with RECORDTYPE='SEGMEMTED'. 4. If the file uses a VAX floating-point format, specify the appropriate VAX format by using one of the methods described in Converting Unformatted Numeric Data.
l
Be aware that Fortran-only programs are usually easier to port than mixedlanguage programs.
See the Compatibility appendix and other sections of the appropriate Compaq Fortran for Alpha systems user manual (Compaq Fortran User Manual for Tru64 UNIX and Linux Systems or the Compaq Fortran User Manual for OpenVMS Alpha Systems).
.f, .for, .f90 source file types # Constants constants using other than base 10
Compatibility Information
l l l l
Page 14 of 16
l l l l l l l
l l l l l l
l l l l l l l l l l l l l
l l l l l l l l l
C strings NULL terminated strings MBCS characters in comments MBCS characters in string literals Conditional compilation and metacommand (directive) expressions ($DEFINE, $UNDEFINE, $IF, $ELSEIF, $ELSE, $ENDIF) !MS$ directive form (see Compiler Directives: table) $FREEFORM, $NOFREEFORM, $FIXEDFORM source file format $OBJCOMMENT place library-search record in object file $INTEGER, $REAL selects size $FIXEDFORMLINESIZE line length for fixed-form source $STRICT, $NOSTRICT F90 conformance $ATTRIBUTES, identifier attributes (C, STDCALL, REFERENCE, VALUE, DLLIMPORT, DLLEXPORT, EXTERN, ALIAS, VARYING) $PACK structure packing Kind numbers match bytes kind parameters AUTOMATIC attribute automatic storage class Integer Pointers (Cray pointers) VAX Structures equivalent to Fortran 90 sequence derived types Mixing logicals and numerics logicals used with arithmetic operators and variables Argument matching for procedure calls Mixing integer kinds to intrinsics Byte data type equivalent to INTEGER*1 $ATTRIBUTES [ ] Form $ATTRIBUTES ALIAS external name for a subprogram $ATTRIBUTES C, STDCALL calling and naming conventions $ATTRIBUTES VALUE, REFERENCE argument passing calling conventions $ATTRIBUTES DLLIMPORT, DLLEXPORT import from/export to DLL Character and non-character equivalence Double complex data type .XOR. exclusive disjunction Integer arguments in logical expressions OPEN statement specifier options: BLOCKSIZE= internal buffer size used in I/O CARRIAGECONTROL= controls the output of formatted files MODE= controls access to files on networked systems TITLE= affects and IOFOCUS= controls QuickWin child windows SHARE= controls simultaneous access to files on networked systems Default carriage control Implicit open prompt user for filenames Special device names for FILE in OPEN statements FORM=BINARY in INQUIRE/OPEN statements Unformatted sequential file form Q edit descriptor number of characters remaining in the input record \ descriptor prevents writing an end-of-record mark $ edit descriptor suppresses the carriage return at the end of a record X edit descriptor default 1
Compatibility Information
l l
Page 15 of 16
l l l
l l l l l l l l l l l
l l l l l l l l l l l l l l l l
l l l l l l l l
Ew.dDe and Gw.dDe edit descriptors similar to Ew.dEe and Gw.dEe Variable Format Expressions (VFEs) integer expression in FORMAT statement Expanded missing ,'s in FORMAT statements optional commas Expanded namelist start/end sequences All path names, including driver, compiler, and INCLUDE statements are MBCS enabled [not on Windows 95] UNC pathnames Long filenames 7200 character statement length Free form infinite line length $DECLARE and $NODECLARE like IMPLICIT NONE Logical truth: 0 = false, non-zero = true $ATTRIBUTES EXTERN variable allocated in another source file $ATTRIBUTES VARYING variable number of arguments Alternate PARAMETER syntax no parenthesis $ in identifiers INTERFACE TO subroutine/function prototype, however global scoping is not supported Argument passing modifiers %VAL, %REF Argument passing modifiers %DESCR (treated as %REF) CRAY pointer support for procedure names (for COM/OLE support) $ATTRIBUTES ALLOCATABLE allocatable array Mixing subroutines/functions in generic interfaces $MESSAGE output message during compilation Listing directives $TITLE, $SUBTITLE STATIC attribute static storage class EOF checks for end of file LOC equivalent to %LOC HFIXconverts to short integer INT1 converts to one byte integer by truncating INT2 converts to two byte integer by truncating INT4 converts to four byte integer by truncating JFIX same as INT4 MALLOC allocates a memory block of the specified number of bytes and returns an integer pointer to the block FREE frees the memory block specified by the integer pointer COTAN returns cotangent DCOTAN returns double precision cotangent IMAG returns the imaginary part of complex number IBCHNG reverses value of bit ISHA shifts arithmetically left or right ISHC performs a circular shift ISHL shifts logically left or right
The following known source incompatibilities exist between Microsoft Fortran PowerStation Version 4 and Visual Fortran:
Compatibility Information
Page 16 of 16
l l l
l l l l l
DATA statement style initialization in attribute style declaration (not supported) Debug lines (other than D) (not supported) $OPTIMIZE change optimization options (not supported) Listing directives $PAGE, $PAGESIZE, $LINESIZE, $[NO]LIST, $INCLUDE (not supported) $DEBUG, $NODEBUG additional run-time checking (not supported) $LINE = C's #line (not supported) Internal files can be any type (not supported) Negative I/O unit numbers (not supported) Interface blocks using INTERFACE [TO] at the beginning of a source file to provide global scoping for subsequent program units (not supported) Visual Fortran uses standard Fortran 90 semantic rules about interface block placement and use. Tab continuation lines that start with characters other than digits 1 through 9 (not supported)
Page 1 of 79
Overview of Visual Fortran Tools Using Tools From the Command Line Setting Up the Command Console Fortran Compiler and Linker MS-DOS Editor Building Projects with NMAKE Resource Compiler Options Managing Libraries with LIB Editing Files with EDITBIN Examining Files with DUMPBIN Editing Format Descriptors with the Format Editor Profiling Code from the Command Line Fortran Tools: FSPLIT and FPR
Integrated Tools in the Visual Development Environment Editor Provides general editing functionality. It recognizes Fortran syntax and can be customized. For more information, see "Text Editor" in the Visual C++ User's Guide. Provides general debug functionality. For more information, see Debugging Fortran Programs Presents format code with resulting data layout. For more information, see Editing Format Descriptors with the Format Editor. Simplifies the use of Component Object Model (COM) and Automation (OLE Automation) objects with Fortran. For more information, see Using COM and Automation Objects.
Using Visual Fortran Tools Profiler (PROFILE, PLAITS, and PREP) Resource Editors
Page 2 of 79
Determines unexecuted code or indicates where an application is spending most of its time. For more information, see Profiling Code from the Command Line. Develops user-interface components for projects; for example, to build a dialog box. For more information, see Using Dialogs and see "Resource Editors" in the Visual C++ User's Guide. Creates an information file with details about the symbols in your program. The browse window displays this information and lets you move among instances of the symbols in your source code. For more information, see Source Browser Information for a Configuration.
Additional Tools1 Linker (LINK) Lets you link object files and libraries, creating 32-bit executable images or DLLs. For more information, see Using the Compiler and Linker from the Command Line and Compiler and Linker Options. Lets you manage object libraries, create import libraries to reference exported symbol definitions used when you build Dynamic Link Libraries (DLLs), and extract library members. For more information, see Managing Libraries with LIB. Displays various information from .obj, .exe, and .libs files. For more information, see Examining Files with DUMPBIN. Lets you modify execution characteristics of a program. For more information, see Editing Files with EDITBIN. Lets you view the binary representation of real numbers in single and double format. For more information, see Viewing Floating-Point Representations with BitViewer. Lets you convert binary resource files (.res) to linkable object (.obj) files. For more information, see CVTRES below. Lets you monitor Dynamic Data Exchange (DDE) activity between processes. For more information, see "Windows Utilities" in the Visual C++ User's Guide.
Librarian (LIB)
Microsoft Binary File Dumper (DUMPBIN) Microsoft Binary File Editor (EDITBIN) BitViewer (BITVIEW) CVTRES
DDESpy (DDESPY)
Page 3 of 79
Lets you preprocess Fortran files; similar to the C preprocessor (CPP). For more information, see /fpp. Lets you transform files formatted according to Fortran's carriage control conventions into files formatted according to line printer conventions. For more information, see Fortran Tools: FSPLIT and FPR. Lets you split a multi-routine Fortran file into individual files. FSPLIT works on FORTRAN 77 files, while FSPLIT90 works on Fortran 90 files. For more information, see Fortran Tools: FSPLIT and FPR. Lets you build projects based on commands contained in a description (makefile) file. For more information, see Building Projects with NMAKE. Lets you browse, configure, test, and activate any COM class on your system; also called the OLEViewer. For more information, see OLE Object Viewer below. Lets you examine and modify processes and threads running on your system. For more information, see "Windows Utilities" in the Visual C++ User's Guide. Compiles various resources so they can be included in an image. For more information, see Resource Compiler Command Line. Lets you view the contents of the OLE Running Object Table. For more information, see Running Object Table Viewer below. Lets you monitor windows messages. For more information, see "Windows Utilities" in the Visual C++ User's Guide. Lets you graphically compare the contents of two files or two directories. For more information, see "Windows Utilities" in the Visual C++ User's Guide. Lets you capture and enlarge an area of the Windows desktop. For more information, see "Windows Utilities" in the Visual C++ User's Guide.
Microsoft Program Maintenance Utility (NMAKE) OLE Object Viewer (OLEVIEW) PView (PVIEW)
Resource Compiler (RC) Running Object Table Viewer (IROTVIEW) Spy++ (SPYXX)
To access these tools from a command window, the Visual Fortran environment must be initialized, as described in "Using the Command-Line
Using Visual Fortran Tools Interface" in Compaq Visual Fortran Installing and Getting Started. Miscellaneous Tool Information
Page 4 of 79
This section briefly describes tools that are not described in detail elsewhere in the documentation. CVTRES Binary resource files (.res) cannot be linked. CVTRES lets you convert a binary resource file into a linkable object file (.obj). For example:
cvtres /out:test.obj test.res
Running Object Table Viewer (IROTVIEW) The Running Object Table Viewer lets you view the contents of the OLE Running Object Table (ROT). This table contains information about ActiveX and OLE objects currently existing in memory. OLE Object Viewer (OLEVIEW) The OLE/COM Object Viewer (OLEViewer) lets you do the following:
l
l l
Browse, in a structured way, all of the Component Object Model (COM) classes installed on your machine. See the registry entries for each class in an easy-to-read format. Configure any COM class (including JavaTM-based classes) on your system. This includes Distributed COM activation and security settings. Configure system-wide COM settings, including enabling or disabling Distributed COM. Test any COM class by double-clicking its name. The list of interfaces that class supports will be displayed. Double-clicking an interface entry allows you to invoke a viewer that will "exercise" that interface. Activate COM classes locally or remotely. Use this to test Distributed COM setups. View type library contents. Use this to figure out what methods, properties, and events an ActiveX Control supports. Copy a properly formatted OBJECT tag to the clipboard for inserting into an HTML document.
The OLEViewer supports plug-in interface viewers. The code for the interface viewers is included in OLEView (in IVIEWERS.DLL).
Page 5 of 79
environment called Microsoft visual development environment, you can still use many software tools directly from the command line. If you prefer to use a text-based environment, you can build your programs or libraries in the console (such as the Fortran Command Prompt in the Compaq Visual Fortran program folder), a command-line operating environment similar to MS-DOS provided by your Windows operating system. However, to get the benefit of components that you cannot use from the command line, you may want to do some of your work from the console, and some of it in the visual development environment. When you run an application for Windows (such as the Format Statement Editor) from the command line, Windows recognizes that the program does not execute within the command window and acts accordingly. You can tell Windows to run a program with its own resources by using the START command. For example, to run the Library Manager as a separate task, the command is:
START LIB.EXE
Visual Fortran contains an extensive electronic reference you view with the HTML Help Viewer. This includes the Visual Fortran online documentation and a search engine. To access HTML Help Viewer books from outside the visual development environment, click on the Online Documentation item in the Visual Fortran program folder. If you want to use Visual Fortran from the command line, you can still use the visual development environment to display HTML Help Viewer, and task switch between it and the console. The following related sections discuss command-line tools:
l l l l l l l l l l l
Setting Up the Command Console Fortran Compiler and Linker MS-DOS Editor Building Projects with NMAKE Resource Compiler Options Managing Libraries with LIB Editing Files with EDITBIN Examining Files with DUMPBIN Editing Format Descriptors with the Format Editor Profiling Code from the Command Line Fortran Tools: FSPLIT and FPR
For a summary of all Visual Fortran tools, see Overview of Visual Fortran Tools.
Page 6 of 79
The console window provides a similar working environment to that provided by running a version of MS-DOS instead of the Windows operating system. You can use any command recognized by MS-DOS in the Windows NT 4 or Windows 2000 console, plus some additional commands. Because the command console runs within the context of Windows, you get the additional benefit that you can easily switch between the command console and other applications for Windows. If you want, you can even have multiple instances of the command console open at once. When you are finished working in a command console window, use the EXIT command to close the window and end the session. To start the command console window provided by your operating system, open the Start menu and select MS-DOS Prompt from the Programs submenu. For more information:
l l
Configuring the Command Console Window Setting Search Paths in the Console
l l
Whether the command console takes over the entire screen or is presented in a window The typeface and type size used to display text in the command console The size of the command console text buffer and the position of the command console window if it is presented in a window The colors used to display text in the command console
Page 7 of 79
The size of the command history buffer used to store commands that scroll out of view The amount of each type of memory that is available to programs running in the command console Special configuration files to be run when the console session begins
The controls that you use to make these adjustments depend upon which version of Windows you are using. The operating system provides a way to specify configuration settings for all subsequent sessions with the command console. On Windows NT 4 and Windows 2000 systems, use the Control Panel. On Windows 98, Windows Me, and Windows 95 systems, use the Properties dialog box to set all of the initial and operating conditions for the command console. With the command window open, do the following: 1. Click the right mouse button at the top of the window. A pop-up menu appears. 2. Click Properties. 3. From the Properties dialog, set up the console display as you like.
Each time you begin a session on Windows 98 or Windows 95 systems, by specifying it in the Program tab of the Properties dialog box for the console icon.
Page 8 of 79
On Windows NT 4 and Windows 2000 systems, you can specify it as the initialization file with the PIF Editor.
The instructions specify the PATH, INCLUDE, and LIB environment variables. For example, the lines in the batch file that sets the INCLUDE environment variable include:
set LIB=%DFcdrom%\DF98\LIB;%DFcdrom%\VC98\LIB;%LIB%
The batch file inserts the directories used by Visual Fortran at the beginning of the existing paths. Because these directories appear first, they are searched before any directories in the path lists provided by Windows. This is especially important if the existing path includes directories with files having the same names as those needed by Visual Fortran. As described in "Using the Command-Line Interface" in Compaq Visual Fortran Installing and Getting Started, the Fortran Command Prompt window sets these variables for you automatically. To activate this command window, select the Fortran Command Prompt icon in the Compaq Visual Fortran program folder.
Using the Compiler and Linker from the Command Line Compiler and Linker Options
MS-DOS Editor
You can use the MS-DOS Editor (EDIT.EXE) or any text editor to create your source programs, but you will not be able to perform the specialized functions built into the visual development environment such as multi-file searches, and your source code will not be displayed with syntax coloring. You invoke the MS-DOS Editor by typing EDIT followed by the name of the file you want to edit. For example:
EDIT test.f90
Page 9 of 79
Some projects require an extensive set of build instructions to ensure that each component is built with the appropriate options. With the Microsoft visual development environment, you can specify build instructions by source file, and you can have separate sets of instructions for the debug and release builds of a project. In the visual development environment, you select these options in a set of dialog boxes. For information on creating (exporting) a makefile from the visual development environment, see The Project Makefile. When you build projects from the command line, you can put your build instructions into a special build file, and run the build process with NMAKE, the Microsoft Program Maintenance Utility. Other command-line building methods include using indirect command files (see DF Indirect Command File Use) and .BAT files. The Microsoft Program Maintenance Utility (NMAKE.EXE) is a 32-bit tool that builds projects based on commands contained in a description file. This section discusses the following:
l l l l l l l l l
Running NMAKE Contents of a Makefile Description Blocks Commands in a Makefile Inline Files in a Makefile Macros and NMAKE NMAKE Inference Rules Dot Directives Makefile Preprocessing
Running NMAKE
The syntax for NMAKE is: NMAKE [option...] [macros...] [targets...] [@commandfile...] NMAKE builds only specified targets or, if none is specified, the first target in the makefile is used. The first makefile target can be a pseudotarget (a label used in place of a filename in a dependency line) that builds other targets. NMAKE uses makefiles specified with the /F option. If /F is not specified, it uses the MAKEFILE file in the current directory. If no makefile is specified, it uses inference rules to build command-line targets. The commandfile text file contains command-line input. Other input can precede or follow @commandfile. A path is permitted. In commandfile, line breaks are treated as spaces. Enclose macro definitions in quotation marks if they contain spaces.
Page 10 of 79
On targets, see Description Blocks On macros, see Macros and NMAKE On options, see NMAKE Options
NMAKE Options
NMAKE options are described in the following sections. Options are preceded by either a slash ( / ) or a dash ( - ) and are not case sensitive. Use ! CMDSWITCHES (described in Makefile Preprocessing Directives) to change option settings in a makefile or in TOOLS.INI. This section describes the following topics:
l l l
NMAKE Option Descriptions TOOLS.INI and NMAKE Exit Codes from NMAKE
/D
/E
Page 11 of 79
The option /F filename specifies filename as a makefile. Spaces or tabs can precede filename. Specify the /F option once for each makefile. To supply a makefile from standard input, specify a dash (-) for filename. End keyboard input with either F6 or Ctrl+Z. The option /HELP or /? displays a brief summary of NMAKE command-line syntax. Ignores exit codes from all commands. To set or clear the /I option for part of a makefile, use !CMDSWITCHES (described in Makefile Preprocessing Directives). To ignore exit codes for part of a makefile, use a dash (-) command modifier or .IGNORE. Overrides the /K option if both are specified. Continues building unrelated dependencies if a command returns an error; also issues a warning and returns an exit code of 1. By default, NMAKE halts if any command returns a nonzero exit code. Warnings from the /K option are suppressed by the /C option; the /I option overrides the /K option if both are specified. Displays but does not execute commands; preprocessing commands are executed. Does not display commands in recursive NMAKE calls. Useful for debugging makefiles and checking timestamps. To set or clear the /N option for part of a makefile, use !CMDSWITCHES (described in Makefile Preprocessing Directives). Suppresses the NMAKE copyright message. Displays information (macro definitions, inference rules, targets, .SUFFIXES list) to standard output, then runs the build. If no makefile or command-line target exists, it displays information only. Use with the /D option to debug a makefile. Checks timestamps of targets; does not run the build. Returns a zero exit code if all are up-to-date and a nonzero exit code if any target is not. Preprocessing commands are executed. Useful when running NMAKE from a batch file. Clears the .SUFFIXES list and ignores inference rules and macros that are defined in the TOOLS.INI file or that are predefined. Suppresses display of executed commands. To suppress display in part of a makefile, use the @ command modifier or .SILENT. To set or clear the /S option for part of a makefile, use ! CMDSWITCHES (described in Makefile Preprocessing Directives).
/HELP, /? /I
/K
/N
/NOLOGO /P
/Q
/R /S
Page 12 of 79
Updates timestamps of command-line targets (or first makefile target) and executes preprocessing commands but does not run the build. The option /X filename sends NMAKE error output to filename instead of standard error. Spaces or tabs can precede filename. To send error output to standard output, specify a dash (-) for filename. Does not affect output from commands to standard error. Disables batch-mode inference rules. When this option is selected, all batch-mode inference rules are treated as regular inference rules.
/X filename
/Y
A syntax error in the makefile An error or exit code from a command An interruption by the user
4 255
System error out of memory Target is not up-to-date (issued only when the /Q option is used)
Page 13 of 79
Contents of a Makefile
A makefile contains:
l l l l l l
Description blocks Commands Macros Inference Rules Dot Directives Preprocessing Directives
Other features of a makefile include wildcards, long filenames, comments, and special characters.
Comments in a Makefile
Precede a comment with a number sign (#). NMAKE ignores text from the number sign to the next newline character. The following are examples of comments:
# Comment on line by itself OPTIONS = /MAP # Comment on macro definition line all.exe : one.obj two.obj # Comment on dependency line link one.obj two.obj # Comment in commands block # copy *.obj \objects # Command turned into comment copy one.exe \release .obj.exe: # Comment on inference rule line link $< my.exe : my.obj ; link my.obj # Error: cannot comment this # Error: # must be the first character .obj.exe: ; link $< # Error: cannot comment this
Page 14 of 79
Description Blocks
A description block is a dependency line optionally followed by a commands block:
targets... : dependents... commands...
A dependency line specifies one or more targets and zero or more dependents. A target must be at the start of the line. Separate targets from dependents by a colon (:); spaces or tabs are allowed. To split the line, use a backslash ( \ ) after a target or dependent. If a target does not exist, has an earlier timestamp than a dependent, or is a pseudotarget, NMAKE executes the commands. If a dependent is a target elsewhere and does not exist or is out-of-date with respect to its own dependents, NMAKE updates the dependent before updating the current dependency. For more information:
l l l l
Page 15 of 79
Targets
In a dependency line, specify one or more targets, using any valid filename or pseudotarget. Separate multiple targets with one or more spaces or tabs. Targets are not case sensitive. Paths are permitted with filenames. A target cannot exceed 256 characters. If the target preceding the colon is a single character, use a separating space; otherwise, NMAKE interprets the letter-colon combination as a drive specifier.
Pseudotargets
A pseudotarget is a label used in place of a filename in a dependency line. It is interpreted as a file that does not exist and so is out-of-date. NMAKE assumes a pseudotarget's timestamp is the most recent of all its dependents; if it has no dependents, the current time is assumed. If a pseudotarget is used as a target, its commands are always executed. A pseudotarget used as a dependent must also appear as a target in another dependency; however, that dependency does not need to have a commands block. Pseudotarget names follow the filename syntax rules for targets. However, if the name does not have an extension (that is, does not contain a period), it can exceed the 8-character limit for filenames and can be up to 256 characters long.
Multiple Targets
NMAKE evaluates multiple targets in a single dependency as if each were specified in a separate description block as follows: This...
bounce.exe leap.exe : jump.obj echo Building...
Cumulative Dependencies
Dependencies are cumulative in a description block if a target is repeated as follows:
Page 16 of 79
This...
bounce.exe : jump.obj bounce.exe : up.obj echo Building bounce.exe...
Multiple targets in multiple dependency lines in a single description block are evaluated as if each were specified in a separate description block, but targets that are not in the last dependency line do not use the commands block as follows: This...
bounce.exe leap.exe : jump.obj bounce.exe climb.exe : up.obj echo Building...
Dependency Side Effects If a target is specified with a colon (:) in two dependency lines in different locations, and if commands appear after only one of the lines, NMAKE interprets the dependencies as if adjacent or combined. It does not invoke an inference rule for the dependency that has no commands, but instead assumes that the dependencies belong to one description block and executes the commands specified with the other dependency as follows:
Page 17 of 79
This...
bounce.exe : jump.obj echo Building bounce.exe... bounce.exe : up.obj
Dependents
In a dependency line, specify zero or more dependents after the colon (:) or double colon (::) using any valid filename or pseudotarget. Separate multiple dependents with one or more spaces or tabs. Dependents are not case sensitive. Paths are permitted with filenames. Inferred Dependents An inferred dependent is derived from an inference rule and is evaluated before explicit dependents. If an inferred dependent is out-of-date with respect to its target, NMAKE invokes the commands block for the dependency. If an inferred dependent does not exist or is out-of-date with respect to its own dependents, NMAKE first updates the inferred dependent. For more information, see NMAKE Inference Rules. Search Paths for Dependents Each dependent has an optional search path, specified as follows: {directory[;directory...]}dependent NMAKE looks for a dependent first in the current directory, and then in directories in the order specified. A macro can specify part or all of a search path. Enclose directory names in braces ({ }); separate multiple directories with
Using Visual Fortran Tools a semicolon (;). No spaces or tabs are allowed.
Page 18 of 79
Commands in a Makefile
A description block or inference rule specifies a block of commands to run if the dependency is out-of-date. NMAKE displays each command before running it, unless the /S option, .SILENT, !CMDSWITCHES, or @ is used. NMAKE looks for a matching inference rule if a description block is not followed by a commands block. A commands block contains one or more commands, each on its own line. No blank line can appear between the dependency or rule and the commands block. However, a line containing only spaces or tabs can appear; this line is interpreted as a null command and no error occurs. Blank lines are permitted between command lines. A command line begins with one or more spaces or tabs. A backslash ( \ ) followed by a newline character is interpreted as a space in the command; use a backslash at the end of a line to continue a command onto the next line. NMAKE interprets the backslash literally if any other character, including a space or tab, follows the backslash. A command preceded by a semicolon (;) can appear on a dependency line or inference rule, whether or not a commands block follows:
project.obj : project.f90 ; df /c project.f90
Page 19 of 79
Turns off error checking for command. By default, NMAKE halts when a command returns a nonzero exit code. If -number is used, NMAKE stops if the exit code exceeds number. Spaces or tabs cannot appear between the dash and number; at least one space or tab must appear between number and command. Use the /I option to turn off error checking for the entire makefile; use .IGNORE to turn off error checking for part of the makefile. Executes command for each dependent file if command uses $** (all dependent files in the dependency) or $? (all dependent files in the dependency with a later timestamp than the target).
!command
Page 20 of 79
l l l l
Specifying an Inline File in Makefiles Creating Inline File Text Reusing Inline Files in Makefiles Multiple Inline Files
Page 21 of 79
Defining an NMAKE Macro Special Characters in NMAKE Macros Null and Undefined NMAKE Macros Where to Define Macros Precedence in Macro Definitions Using an NMAKE Macro Macro Substitution Special NMAKE Macros
Page 22 of 79
The macroname is a combination of letters, digits, and underscores ( _ ) up to 1024 characters, and is case sensitive. The macroname can contain an invoked macro. If macroname consists entirely of an invoked macro, the macro being invoked cannot be null or undefined. The string can be any sequence of zero or more characters. A null string contains zero characters or only spaces or tabs. The string can contain a macro invocation.
Page 23 of 79
Use the /E option to cause macros inherited from environment variables to override makefile macros with the same name. Use !UNDEF to override a command line.
Macro Substitution
To substitute text within a macro, use the following syntax: $(macroname:string1=string2) When macroname is invoked, each occurrence of string1 in its definition string is replaced by string2. Macro substitution is case sensitive and is literal; string1 and string2 cannot invoke macros. Substitution does not modify the original definition. You can substitute text in any predefined macro except $$@. No spaces or tabs precede the colon; any after the colon are interpreted as literal. If string2 is null, all occurrences of string1 are deleted from the macro's definition string.
Page 24 of 79
Filename Macros Recursion Macros Command Macros and Options Macros Environment-Variable Macros
Filename Macros Filename macros are predefined as filenames specified in the dependency (not full filename specifications on disk). These macros do not need to be enclosed in parentheses when invoked; specify only a $ as shown. Macro $@ $$@ $* $** $? $< Meaning Current target's full name (path, base name, extension), as currently specified. Current target's full name (path, base name, extension), as currently specified. Valid only as a dependent in a dependency. Current target's path and base name minus file extension. All dependents of the current target. All dependents with a later timestamp than the current target. Dependent file with a later timestamp than the current target. Valid only in commands in inference rules.
To specify part of a predefined filename macro, append a macro modifier and enclose the modified macro in parentheses. Modifier D B F R Resulting Filename Part Drive plus directory Base name Base name plus extension Drive plus directory plus base name
Recursion Macros
Page 25 of 79
Use recursion macros to call NMAKE recursively. Recursive sessions inherit command-line and environment-variable macros. They do not inherit makefiledefined inference rules or .SUFFIXES and .PRECIOUS specifications. To pass macros to a recursive NMAKE session, either set an environment variable with the SET command before the recursive call or define a macro in the command for the recursive call. Macro MAKE MAKEDIR MAKEFLAGS Definition Command used originally to invoke NMAKE. Current directory when NMAKE was invoked. Options currently in effect. Use as /$(MAKEFLAGS).
Command Macros, Options Macros Command macros are predefined for Microsoft products. Options macros represent options to these products and are undefined by default. Both are used in predefined inference rules and can be used in description blocks or userdefined inference rules. Command macros can be redefined to represent part or all of a command line, including options. Options macros generate a null string if left undefined. Product Macro Assembler Basic Compiler C Compiler COBOL Compiler C++ Compiler C++ Compiler Compaq Visual Fortran Compiler Pascal Compiler Resource Compiler Command Macro AS BC CC COBOL CPP CXX FOR PASCAL RC Defined as: ml bc cl cobol cl cl df pl rc Options Macro AFLAGS BFLAGS CFLAGS COBFLAGS CPPFLAGS CXXFLAGS FFLAGS PFLAGS RFLAGS
Page 26 of 79
Environment-Variable Macros NMAKE inherits macro definitions for environment variables that exist before the start of the session. If a variable was set in the operating-system environment, it is available as an NMAKE macro. Use the /E option to cause macros inherited from environment variables to override any macros with the same name in the makefile. Environment-variable macros can be redefined in the session, and this changes the corresponding environment variable. You can also change environment variables with the SET command. Using the SET command to change an environment variable in a session does not change the corresponding macro, however. For example:
PATH=$(PATH);\nonesuch all: echo %PATH%
In this example, changing PATH changes the corresponding environment variable PATH; it appends \nonesuch to your path. If an environment variable is defined as a string that would be syntactically incorrect in a makefile, no macro is created and no warning is generated. If a variable's value contains a dollar sign ($), NMAKE interprets it as the beginning of a macro invocation. Using the macro can cause unexpected behavior.
Page 27 of 79
NMAKE may invoke a rule for an inferred dependent even if an explicit dependent is specified. For more information:
l l l l l l
Defining an Inference Rule in NMAKE Search Paths in Inference Rules Batch-Mode Rules Predefined Inference Rules Inferred Dependents and Rules Precedence in NMAKE Inference Rules
Batch-Mode Rules
Page 28 of 79
Batch-mode inference rules provide only one invocation of the inference rule when N commands go through this inference rule. Without batch-mode inference rules, it would require N commands to be invoked. N is the number of dependents that trigger the inference rule. Makefiles that contain batch-mode inference rules must use NMAKE version 1.62 or higher. To check the NMAKE version, run the _NMAKE_VER macro available with NMAKE version 1.62 or higher. This macro returns an integer representing the NMAKE version. For example, the macro returns 162 for NMAKE version 1.62. A batch-mode inference rule has the following syntax: {frompath}.fromext{topath}.toext:: commands The only syntactical difference from the standard inference rule is that the batch-mode inference rule is terminated with a double colon (::). Note: The tool being invoked must be able to handle multiple files. The batch-mode inference rule must use $< as the macro to access dependent files. The batch-mode inference rules can speed up the build process. It is faster to supply files to the compiler in batch, because the compiler driver is invoked only once.
Using Visual Fortran Tools .cxx.obj .bas.obj .cbl.exe .cbl.obj .for.exe .f90.exe .f.exe .for.obj .f90.obj .f.obj .pas.exe .pas.obj .rc.res $(CXX) $(CXXFLAGS) /c $*.cxx $(BC) $(BFLAGS) $*.bas; $(COBOL) $(COBFLAGS) $*.cbl, $*.exe; $(COBOL) $(COBFLAGS) $*.cbl; $(FOR) $(FFLAGS) $*.for $(FOR) $(FFLAGS) $*.f90 $(FOR) $(FFLAGS) $*.f $(FOR) /c $(FFLAGS) $*.for $(FOR) /c $(FFLAGS) $*.f90 $(FOR) /c $(FFLAGS) $*.f $(PASCAL) $(PFLAGS) $*.pas $(PASCAL) /c $(PFLAGS) $*.pas $(RC) $(RFLAGS) /r $* cl /c $*.cxx bc $*.bas;
Page 29 of 79
cobol $*.cbl, $*.exe; cobol $*.cbl; fl32 $*.for fl32 $*.f90 fl32 $*.f fl32 $*.for /c fl32 $*.f90 /c fl32 $*.f /c pl $*.pas pl /c $*.pas rc /r $*
toext matches the target's extension. fromext matches the extension of a file that has the target's base name and that exists in the current or specified directory. fromext is in .SUFFIXES; no other fromext in a matching rule has a higher .SUFFIXES priority. No explicit dependent has a higher .SUFFIXES priority.
Inferred dependents can cause unexpected side effects. If the target's description block contains commands, NMAKE executes those commands and not the commands in the rule.
Page 30 of 79
2. An inference rule defined in Tools.ini; later definitions have precedence. 3. A predefined inference rule.
.PRECIOUS : targets
.SILENT :
.SUFFIXES : list
To change the .SUFFIXES list order or to specify a new list, clear the list and specify a new setting. To clear the list, specify no extensions after the colon:
.SUFFIXES :
Page 31 of 79
spaces or tabs. To see the current setting of .SUFFIXES, run NMAKE with the /P option.
Makefile Preprocessing
You can control the NMAKE session by using preprocessing directives and expressions. Preprocessing instructions can be placed in the makefile. Using directives, you can conditionally process your makefile, display error messages, include other makefiles, undefine a macro, and turn certain options on or off. For more information:
l l l l
Makefile Preprocessing Directives Expressions in Makefile Preprocessing Makefile Preprocessing Operators Executing a Program in Preprocessing
!CMDSWITCHES {+ | -}option... Turns each option listed on or off. Spaces or tabs must appear before the + or - operator; none can appear between the operator and the option letters. Letters are not case sensitive and are specified without a slash (/). To turn some options on and others off, use separate specifications of ! CMDSWITCHES. Only /D, /I, /N, and /S can be used in a makefile. In Tools.ini, all options are allowed except /F, /HELP, /NOLOGO, /X, and /?. Changes specified in a description block do not take effect until the next description block. This directive updates the MAKEFLAGS recursion macro; changes are inherited during recursion if MAKEFLAGS is specified.
!ERROR text Displays text in error U1050, then halts NMAKE, even if /K, /I, .IGNORE, ! CMDSWITCHES, or the dash (-) command modifier is used. Spaces or tabs before text are ignored.
!MESSAGE text Displays text to standard output. Spaces or tabs before text are ignored.
Page 32 of 79
!INCLUDE [<]filename[>] Reads filename as a makefile, then continues with the current makefile. NMAKE searches for filename first in the specified or current directory, then recursively through directories of any parent makefiles, then, if filename is enclosed by angle brackets (<>), in directories specified by the INCLUDE macro, which is initially set to the INCLUDE environment variable. Useful to pass .SUFFIXES settings, .PRECIOUS, and inference rules to recursive makefiles.
!IF constantexpression Processes statements between !IF and the next !ELSE or !ENDIF if constantexpression evaluates to a nonzero value.
!IFDEF macroname Processes statements between !IFDEF and the next !ELSE or !ENDIF if macroname is defined. A null macro is considered to be defined.
!IFNDEF macroname Processes statements between !IFNDEF and the next !ELSE or !ENDIF if macroname is not defined.
!ELSE [IF constantexpression | IFDEF macroname | IFNDEF macroname] Processes statements between !ELSE and the next !ENDIF if the prior !IF, ! IFDEF, or !IFNDEF statement evaluated to zero. The optional keywords give further control of preprocessing.
!ENDIF Marks the end of an !IF, !IFDEF, or !IFNDEF block. Any text after !ENDIF on
Page 33 of 79
Using Visual Fortran Tools | ^ && Bitwise OR Bitwise XOR Logical AND > <= >= Greater than Less than or equal to Greater than or equal to
Page 34 of 79
To compare two strings, use the equality (= =) operator and the inequality (!=) operator. Enclose strings in double quotation marks.
Page 35 of 79
l l l l
Including Resources in an Application Creating a Resource Definition File Resource Compiler Command Line The online Platform SDK sections under "Windows Programming Guidelines," "Platform SDK Tools," such as "Compiling" and "Using the Resource Compiler."
l l
Comments (single-line comments or block line comments) Predefined macros Preprocessing directives, which instruct RC to perform actions on the script before compiling. Preprocessor operators Resource definition statements, which name and describe resources. Statements can be single-line or multiline statements.
Page 36 of 79
The following example shows a script file that defines the resources for an application named Shapes:
#include "SHAPES.H" ShapesCursor CURSOR SHAPES.CUR ShapesIcon ICON SHAPES.ICO BEGIN POPUP "&Shape" BEGIN MENUITEM MENUITEM MENUITEM MENUITEM MENUITEM END END
"&Clear", ID_CLEAR "&Rectangle", ID_RECT "&Triangle", ID_TRIANGLE "&Star", ID_STAR "&Ellipse", ID_ELLIPSE
The CURSOR statement names the application's cursor resource ShapesCursor and specifies the cursor file SHAPES.CUR, which contains the image for that cursor. Custom cursors are not available in Visual Fortran. The ICON statement names the application's icon resource ShapesIcon and specifies the icon file SHAPES.ICO, which contains the image for that icon. The MENU statement defines an application menu named ShapesMenu, a pop-up menu with five menu items. The menu definition, enclosed by the BEGIN and END keywords, specifies each menu item and the menu identifier that is returned when the user selects that item. For example, the first item on the menu, Clear, returns the menu identifier ID_CLEAR when the user selects it. The menu identifiers are defined in the application header file, SHAPES.H. Once you create the resource-definition script (RC) file, use the Resource Compiler Command Line to create the RES file. For details about script files, see the online Platform SDK sections under "Windows Programming Guidelines" and "Platform SDK Tools" such as "Using the Resource Compiler."
Page 37 of 79
contains the names, types, filenames, and descriptions of the resources to be compiled. The options argument can include one or more of the following options: Option /? /d /fo resname /h /i directory /lcodepage Description Displays a list of RC command-line options. Defines a symbol for the preprocessor that you can test with #ifdef. Uses resname for the name of the .RES file. Displays a list of RC command-line options. Causes RC to search the specified directory before searching the directories specified by the INCLUDE environment variable. Specifies default language for compilation. For example, -l409 is equivalent to including the following statement at the top of the resource script file:
LANGUAGE LANG_ENGLISH,SUBLANG_ENGLISH_US
Alternatively, you use #pragma code_page(409) in the .RC file. /n /r /u /v /x Null terminates all strings in the string table. Ignored. Provided for compatibility with existing makefiles. Undefines a symbol. Causes a display of messages that report on the progress of the compiler. Prevents RC from checking the INCLUDE environment variable when searching for header files or resource files.
Options are not case sensitive and a dash (-) can be used in place of a forward slash (/). You can combine single-letter options if they do not require additional arguments. For example, the following commands are equivalent:
RC /V /X SAMPLE.RC rc -vx sample.rc
For more information on these options and the resource compiler, see the online
Page 38 of 79
Platform SDK sections under "Windows Programming Guidelines" and "Platform SDK Tools," such as "Using the Resource Compiler."
Add objects to a library Specify the filename for the existing library and the filenames for the new objects.
Combine libraries Specify the library filenames. You can add objects and combine libraries in a single LIB command.
Replace a library member with a new object Specify the library containing the member object to be replaced and the filename for the new object (or the library that contains it). When an object that has the same name exists in more than one input file, LIB puts the last object specified in the LIB command into the output library. When you replace a library member, be sure to specify the new object or library after the library that contains the old object.
Delete a member from a library Use the /REMOVE option. LIB processes any specifications of /REMOVE after combining all input objects, regardless of command-line order.
Page 39 of 79
Note: You cannot both delete a member and extract it to a file in the same step. You must first extract the member object using /EXTRACT, then run LIB again using /REMOVE. This section describes the Microsoft 32-Bit Library Manager (LIB.EXE). The following topics are covered:
l l l l l
LIB Input/Output Running LIB LIB Options Extracting a Library Member Import Libraries and Export Files
LIB Input/Output
LIB expects types of input files and generates types of output files depending on the mode in which it is used. You can also get information about the resulting library with the /LIST option, and you can examine the contents of the library by using DUMPBIN with the /LINKERMEMBER option. For more information:
l l l l
LIB Input Files LIB Output Files Other LIB Output Viewing Contents of a Library
Note: Object Model Format (OMF) libraries created by the 16-bit version of LIB
Using Visual Fortran Tools cannot be used as input to the 32-bit LIB.
Page 40 of 79
Running LIB
This section presents information on running LIB in any mode. It describes the LIB command line, discusses the use of command files, and gives general rules for using options. For more information:
Page 41 of 79
l l l
Page 42 of 79
spaces or tabs are allowed within an option specification. Use one or more spaces or tabs to separate option specifications on the command line. Option names and their keyword or filename arguments are not case sensitive, but identifiers used as arguments are case sensitive. LIB processes options in the order specified on the command line and in command files. If an option is repeated with different arguments, the last one to be processed takes precedence. The following LIB options apply to all modes of LIB:
l
/NOLOGO Suppresses display of the LIB copyright message and version number and prevents echoing of command files.
/VERBOSE Displays details about the progress of the session. The information is sent to standard output and can be redirected to a file.
Other options apply only to specific modes of LIB. These options are discussed in the sections describing each mode.
LIB Options
The default mode for LIB is to build or modify a library of COFF objects. LIB runs in this mode when you do not specify /EXTRACT (to copy an object to a file) or /DEF (to build an import library). To build a library from objects and/or libraries, use the following syntax: LIB [options...] files... The options that apply to building and modifying a library are listed in the following table: Option /CONVERT Description Converts an import library to the previous (Visual C++ version 5.0) format.
Page 43 of 79
Overrides the environment library path. For more information, see the linker option /LIBPATH. Displays information about the output library to standard output. The output can be redirected to a file. You can use /LIST to determine the contents of an existing library without modifying it. Generates an import library in the previous (Visual C++ version 5.0) format for backwards compatibility. Overrides the default output filename. By default, the output library has the base name of the first library or object on the command line and the extension .LIB. Omits the specified object from the output library. LIB creates an output library by first combining all objects (whether in object files or libraries), then deleting any objects specified with /REMOVE. Tells the operating system how to run a program created by linking to the output library. For more information, see the description of the LINK /SUBSYSTEM option in Compiler and Linker Options.
/LINK50COMPAT /OUT:filename
/REMOVE:object
/SUBSYSTEM
Building an Import Library and Export File Extracting a Library Member Using LIB Options
The files can be COFF object files, 32-bit OMF object files, and existing COFF libraries. LIB creates one library that contains all objects in the specified files. If an input file is a 32-bit OMF object file, LIB converts it to COFF before building the library. LIB cannot accept a 32-bit OMF object that is in a library created by the 16-bit version of LIB. You must first use the 16-bit LIB to extract the object, then you can use the extracted object file as input to the 32-bit LIB. The 16-bit version of LIB is not provided with Visual Fortran. By default, LIB names the output file using the base name of the first object or library file and the extension .LIB. If a file already exists with the same name, the output file overwrites the existing file. To preserve an existing library, use the /OUT option to specify a name for the output file. You can use LIB to perform the following library-management tasks:
l
Page 44 of 79
Specify the filename for the existing library and the filenames for the new objects.
l
Combine libraries Specify the library filenames. You can add objects and combine libraries in a single LIB command.
Replace a library member with a new object Specify the library containing the member object to be replaced and the filename for the new object (or the library that contains it). When an object that has the same name exists in more than one input file, LIB puts the last object specified in the LIB command into the output library. When you replace a library member, be sure to specify the new object or library after the library that contains the old object.
Delete a member from a library Use the /REMOVE option. LIB processes any specifications of /REMOVE after combining all input objects, regardless of command-line order. Note: You cannot both delete a member and extract it to a file in the same step. You must first extract the member object using /EXTRACT, then run LIB again using /REMOVE. This behavior differs from that of the 16-bit LIB (for OMF libraries) provided in some Microsoft products.
Page 45 of 79
a DLL), and it uses the import library to resolve references to those exports in other programs. In most situations, you do not need to use LIB to create your import library. When you link a program (either an executable file or a DLL) that contains exports, LINK automatically creates an import library that describes the exports. Later, when you link a program that references those exports, you specify the import library. However, when a DLL exports to a program that it also imports from, whether directly or indirectly, you must use LIB to create one of the import libraries. When LIB creates an import library, it also creates an export file. You must use the exports file when linking one of the DLLs. For more information, see:
l l
Building an Import Library and Export File Using an Import Library and Export File
cDEC$ ATTRIBUTES DLLEXPORT in one of the objfiles or libraries A specification of /EXPORT:name on the LIB command line A definition in an EXPORTS statement in a deffile
These are the same methods you use to specify exports when linking an exporting program. A program can use more than one method. You can specify parts of the LIB command (such as multiple objfiles or /EXPORT specifications) in a command file in the LIB command, just as you can in a LINK command. The following options apply to building an import library and export file:
l
/DEBUGTYPE:{CV|COFF|BOTH} This option sets the format of debugging information. Specify CV for newstyle Microsoft Symbolic Debugging Information, required by Visual C++ and Visual Fortran. Specify COFF for Common Object File Format (COFF) debugging information. Specify BOTH for both COFF debugging information and old-style Microsoft debugging information.
Page 46 of 79
/OUT:import This option overrides the default output filename for the import library being created. When /OUT is not specified, the default name is the base name of the first object file or library in the LIB command and the extension .LIB. The exports file is given the same base name as the import library and the extension .EXP.
/EXPORT:entryname[=internalname] [,@ordinal[,NONAME]][,DATA] This option exports a function from your program to allow other programs to call the function. You can also export data. Exports are usually defined in a DLL. The entryname is the name of the function or data item as it is to be used by the calling program. You can optionally specify the internalname as the function known in the defining program; by default, internalname is the same as entryname. The ordinal specifies an index into the exports table in the range 1 - 65535; if you do not specify ordinal, LIB assigns one. The NONAME keyword exports the function only as an ordinal, without an entryname.
/INCLUDE:symbol This option adds the specified symbol to the symbol table. This is useful for forcing the use of a library object that otherwise would not be included.
Page 47 of 79
example discussed earlier, if TWO.DLL also exports to ONE.DLL, the import library for TWO.DLL won't exist yet when ONE.DLL is linked. When circular exports exist, you must use LIB to create an import library and exports file for one of the programs. To begin, choose one of the programs on which to run LIB. In the LIB command, list all objects and libraries for the program and specify the /DEF option. If the program uses a .DEF file or /EXPORT specifications, specify these as well. After you create the import library (.LIB) and the export file (.EXP) for the program, you then use the import library when linking the other program or programs. LINK creates an import library for each exporting program it builds. For example, if you ran LIB on the objects and exports for ONE.DLL, you created ONE.LIB and ONE.EXP. You can now use ONE.LIB when linking TWO.DLL; this step also creates the import library TWO.LIB. Finally, link the program you began with. In the LINK command, specify the objects and libraries for the program, the .EXP file that LIB created for the program, and the import library or libraries for the exports used by the program. In the continuing example, the LINK command for ONE.DLL contains ONE.EXP and TWO.LIB, as well as the objects and libraries that go into ONE.DLL. Do not specify the .DEF file and /EXPORT specifications in the LINK command; these are not needed because the exports definitions are contained in the .EXP file. When you link using an .EXP file, LINK does not create an import library because it assumes that one was created when the .EXP file was created.
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 48 of 79
l l
EDITBIN Options
An option consists of an option specifier, which is either a dash (-) or a forward slash (/), followed by the name of the option. Option names cannot be abbreviated. Some options take arguments, specified after a colon (:). No spaces or tabs are allowed within an option specification. Use one or more spaces or tabs to separate option specifications on the command line. Option names and their keyword or filename arguments are not case sensitive. This section discusses the following EDITBIN options:
l l l l l l l l l l l l
/BIND /HEAP /LARGEADDRESSAWARE /NOLOGO /REBASE /RELEASE /SECTION /STACK /SUBSYSTEM /SWAPRUN /VERSION /WS
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 49 of 79
The /BIND option has the following form: /BIND[:PATH=path] Specify the program's executable file and DLLs in the files argument on the EDITBIN command line. The optional path argument to the /BIND option specifies the location of the DLLs used by the specified files. Separate multiple directories with semicolons (;). If path is not specified, EDITBIN searches the directories specified in the PATH environment variable. If path is specified, EDITBIN ignores the PATH variable. By default, the Windows program loader sets the addresses of entry points when it loads a program. The amount of time this takes varies depending on the number of DLLs and the number of entry points referenced in the program. If a program has been modified with the /BIND option, and if the base addresses for the executable file and its DLLs do not conflict with DLLs that are already loaded, the operating system does not need to set these addresses. In a situation where the files are incorrectly based, the operating system will relocate the program's DLLs and recalculate the entry-point addresses; this adds to the program's load time.
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 50 of 79
BASEFILE DOWN
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 51 of 79
This option changes the properties of a section, overriding the properties that were set when the object file for the section was compiled or linked. It has the following form: /SECTION:name[=newname][,properties][,alignment] After the colon ( : ), specify the name of the section. To change the section name, follow name with an equal sign (=) and a newname for the section. To set or change the section's properties, specify a comma (,) followed by one or more property characters. To negate a property, precede its character with an exclamation point (!). The following characters specify memory properties: Property c d e i k m o p r s u w Setting code discardable executable initialized data cached virtual memory link remove link info paged virtual memory read shared uninitialized data write
To control alignment, specify the character "a" followed by a character to set the size of alignment in bytes, as follows:
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 52 of 79
Character 1 2 4 8 p t s x
Specify the properties and alignment characters as a string with no white space. The characters are not case sensitive.
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 53 of 79
NET indicates that the image resides on a network. CD indicates that the image resides on a CD-ROM or similar removable medium.
Use !NET and !CD to reverse the effects of NET and CD.
Use /WS:AGGRESSIVE for applications such as services and screen savers that must have a low impact on the system's memory pool. If the speed of your application matters, do not use /WS:AGGRESSIVE without testing the resulting performance.
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 54 of 79
DUMPBIN Options
An option consists of an option specifier, which is either a dash (-) or a forward slash (/), followed by the name of the option. Option names cannot be abbreviated. Some options take arguments, specified after a colon (:). No spaces or tabs are allowed within an option specification. Use one or more spaces or tabs to separate option specifications on the command line. Option names and their keyword or filename arguments are not case sensitive. Most options apply to all binary files; a few apply only to certain types of files.
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 55 of 79
The DUMPBIN options are as follows: Option /ALL Description Displays all available information except code disassembly. Use the /DISASM option to display disassembly. You can use /RAWDATA:NONE with the /ALL option to omit the raw binary details of the file.
/ARCHIVEMEMBERS Displays minimal information about member objects in a library. /ARCH /DEPENDENTS Dumps the .arch section of an image. Dumps the names of the DLLs from which the image imports functions. Does not dump the names of the imported functions. Dumps the compiler-generated .drective section of an image. Displays disassembly of code sections, using symbols if present in the file. Displays all definitions exported from an executable file or DLL. Displays Frame Pointer Optimization (FPO) records. Displays coff header information. Displays all definitions imported to an executable file or DLL. Output resembles the /EXPORTS option. Displays COFF line numbers. Line numbers exist in an object file if it was compiled with Program Database (/Zi) or Line Numbers Only (/Zd). An executable file or DLL contains COFF line numbers if it was linked with Generate Debug Info (/DEBUG) and COFF Format (/DEBUGTYPE:COFF).
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 56 of 79
The option /LINKERMEMBER[:{1|2}] displays public symbols defined in a library. Specify the 1 argument to display symbols in object order, along with their offsets. Specify the 2 argument to display offsets and index numbers of objects, then list the symbols in alphabetical order along with the object index for each. To get both outputs, specify /LINKERMEMBER without the number argument. The option /OUT:filename specifies a filename for the output. By default, DUMPBIN displays the information to standard output. dumps the exception tables (.pdata) from an image or object. The option /RAWDATA[:{ BYTES | SHORTS | LONGS | NONE }[, number] ] displays the raw contents of each section in the file. The arguments control the format of the display, as follows: Argument Result BYTES The default. Contents are displayed in hexadecimal bytes, and also as ASCII if they have a printed representation. Contents are displayed in hexadecimal words. Contents are displayed in hexadecimal longwords. Raw data is suppressed. This is useful to control the output of the /ALL option. Displayed lines are set to a width that holds number values per line.
/OUT
/PDATA /RAWDATA
Displays any relocations in the object or image. The option /SECTION:section restricts the output to information on the specified section. Displays minimal information about sections, including total size. This option is the default if no other option is specified.
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 57 of 79
Displays the COFF symbol table. Symbol tables exist in all object files. A COFF symbol table appears in an image file only if it is linked with the Generate Debug Info and COFF Format options under Debug Info on the Debug category for the linker (or the /DEBUG and /DEBUGTYPE:COFF options on the command line).
If the line specified by the second parameter is empty, a new format statement is created with the words label FORMAT. To use the Format Editor on a multi-line format statement, the argument list must specify the first line of the format statement. In-line comments in a multiline Format statement are lost when the Format Editor writes the updated format statement and generates new continuation marks. Similarly, the part of a formatted I/O statement that follows the formatting directives is lost when the Format Editor writes the updated directive string back to the file. When you are finished editing the format statement, the Format Editor rewrites the source file with code for the format you have developed. If the file has the extension .F90, the revised code is written with Fortran 90 free-form syntax rules; otherwise, it is written with Fortran 90 fixed-form syntax rules. The Format Editor is installed on the Edit menu during Visual Fortran installation. If you have removed it for any reason, and need to reinstall it, you can do so by choosing Customize from the Tools menu. The argument list, which passes the current file name, line number and column position to the Format Editor, is $File $Line $Column. For more information about adding programs to the Tools menu, select the Help button in the Customize dialog. file://C:\TEMP\~hh10D5.htm 6/21/01
Page 58 of 79
Starting the Format Editor from the Microsoft Visual Development Environment
The Format Editor presents the format code and a sample of the resulting data layout in a window that works like a dialog box. You can edit either the source code or the data layout, and the Format Editor changes the other to match. To open the Format Editor: 1. Load a Fortran source file that contains a FORMAT statement or an I/O edit descriptor. 2. Place the cursor on the first line of the FORMAT statement or on the line containing the edit descriptor. 3. From the Edit menu, choose Format Editor. The Format Editor dialog box opens. The Format Editor dialog box consists of the following text boxes and buttons:
l l l
The edit descriptors in the upper-left text box The sample data display in the lower-left text box The New Field, Remove Field, Change Value, OK, Cancel, and Help buttons along the bottom
When you open the Format editor in a line with an edit descriptor, the editor attempts to parse the first opening quote that precedes the cursor position. When you open the editor in a line containing a FORMAT statement, the editor attempts to parse the edit descriptors in the statement. If the editor is successful, it displays the edit descriptors in the upper-left box and a sample data display in the lower text box. If the editor cannot parse the descriptor string or FORMAT statement, you will get a parse error message. To insert new I/O edit descriptors into existing formatted I/O statements or FORMAT statements in Microsoft visual development environment: 1. 2. 3. 4. Place the cursor in the existing descriptor or FORMAT statement. From the Edit menu, choose Format Editor. Choose New Field from the Format Editor dialog box. Choose the descriptor type (Character, Integer, and so on) and choose whether to insert the descriptor before or after the current descriptor. 5. A default value is used for the descriptor you choose (for example, I5). To change the descriptor value, select the value and type in the new value. (For example, select 5 from I5 and type 8 to get an I8 format.) To insert a new format statement:
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 59 of 79
1. Place the cursor on a blank line. 2. From the Edit menu, choose Format Editor. The Format Editor inserts the words label FORMAT into the file at the cursor. You can define the new edit descriptor with the Format Editor. For a discussion of the features of the Format Editor in Microsoft visual development environment, choose Format Editor from the Edit menu, and click on Help in the Format Editor dialog box.
Profiler Batch Processing Profiler Batch Files Profiler Command-Line Options Exporting Data From the Profiler
Page 60 of 79
Profiling requires three separate programs: PREP, PROFILE, and PLIST. The visual development environment executes all three of these programs for you automatically. To execute them efficiently from the command line, and to customize the output format or specify function and line count profiling, you must write batch files to invoke PREP, PROFILE, and PLIST. You can redirect the output of the batch file to a designated file by using the redirection character (>). A typical profiler batch file might look like this:
PREP /OM /FT /EXC nafxcwd.lib %1 if errorlevel == 1 goto done PROFILE %1 %2 %3 %4 %5 %6 %7 %8 %9 if errorlevel == 1 goto done PREP /M %1 if errorlevel == 1 goto done PLIST /SC %1 >%1.lst :done
Note that the PREP program is called twice - once before the actual profiling and again afterward. The command-line arguments govern PREP's behavior. Intermediate files with extensions .PBI, .PBO, and .PBT are used to transfer information between profiling steps. The first call to PREP generates a .PBI file which is passed to PROFILER. PROFILER generates a .PBO file which is passed in the second call to PREP. The second call to PREP generates a .PBT file which is passed to PLIST. The profiler data flow is shown in the following figure: Profiler Data Flow
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 61 of 79
If the preceding batch file was named FTIME.BAT, and you wanted to profile the program TEST from the command prompt, you would type:
FTIME C:\Program Files\DF98\MYDIR\TEST.EXE
The # character in a response file defines a comment that runs through the end of the line. Five standard batch files ship with the profiler: Filename FTIME.BAT FCOUNT.BAT FCOVER.BAT LCOUNT.BAT LCOVER.BAT Description Function timing Function counting Function coverage Line counting Line coverage
These batch files contain only the minimum parameters for the initial call to file://C:\TEMP\~hh10D5.htm 6/21/01
Page 62 of 79
PREP. Use them as prototypes for your own batch files, which should contain selection parameters. If you ran an unmodified LCOVER batch file for a complex Fortran 90 application with a large number of functions, subroutines and modules, the output report could be thousands of lines long.
PREP
The PREP program runs twice during a normal profiling operation. In Phase I, it reads an .EXE file and then creates .PBI and .PBT files. In Phase II, it reads .PBT and .PBO files and then writes a new .PBT file for PLIST. An 'X' in the following Options table indicates that a PREP command-line option applies to a particular phase. The syntax for PREP is: PREP [options] [programname1] [programname2...programname8] PREP reads the command line from left to right, so the rightmost options override contradictory options to the left. None of the options are case sensitive. You must prefix options with a forward slash ( / ) or a dash ( - ), and options must be separated by spaces. Parameter options programname1 Description Control the kind of profiling, the inclusion and exclusion of code to be profiled, whether to merge profiles, and other profiling features. See the PREP Options table. Filename of primary program to profile (.DBG, .EXE, or .DLL). PROFILE adds the .EXE extension if no extension is given. This parameter must be specified in the first call to PREP and not the second call. Additional programs to profile. These parameters can be specified for the first call to PREP only.
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 63 of 79
PREP Options Phase Option /AT I X II Description Collects attribution data for function timing and function counting. Function attribution reports which function called another function. See the /STACK switch later in this list. Used with function timing, allows you to set the calibrated overhead of profiler calls in the event that your function timing calls have varied because of varied calibrated overhead values. The calibrated overhead is displayed in default (non-tab-delimited) PLIST output. Excludes a specified module from the profile (See the Remarks section). Excludes all modules from the profile (See the Remarks section). Selects function count profiling. Selects function timing profiling. This option causes the profiler to generate count information as well. Selects function coverage profiling. Includes in profile (See the Remarks section). X X Provides a short summary of PREP options. Merges an existing .PBO file (the file generated by PROFILER to be passed in the second call to PREP). Up to eight .PBO files can be merged at a time. The default extension is .PBO. Merges an existing .PBT file (the file generated by the second call to PREP to be passed to PLIST). Up to eight .PBT files can be merged at a time. You cannot merge .PBT files from different profiling methods. The default extension is .PBT. Selects line count profiling.
/CB
X X X X X X X
/IT filename
/LC
file://C:\TEMP\~hh10D5.htm
6/21/01
Using Visual Fortran Tools /LV /M filename /NOLOGO /OI filename /OM X X X X X Selects line coverage profiling. Substitutes for /IT, /IO, and /OT options. Suppresses the PREP copyright message.
Page 64 of 79
Creates a .PBI file (the file generated by the first call to PREP). The default extension is .PBI. If /OI is not specified, the output .PBI file is programname1.PBI. Creates a self-profiling file with _XE or _LL extension for function timing, function counting, and function coverage. Without this option, the executable code is stored in the .PBI file. This option speeds up profiling. X Specifies the output .PBT file. The default extension is .PBT. If /OT is not specified, the output .PBT file is programname1.PBT. Starts profiling with function. The function name must correspond to an entry in the .MAP file. When using the /AT switch, you can also set the stack depth (dpt) to which functions will have their attribution data recorded. X Provides a short summary of PREP options.
X X
Environment Variable The PREP environment variable specifies the default PREP command-line options. If a value for the PREP environment variable is not specified, the default options for PREP are:
/FT /OI filename /OT filename
where filename is set to the programname1 parameter value. Remarks The /INC and /EXC options specify individual .LIB, .OBJ, .FOR and .F90 files. For line counting and line coverage, you can specify line numbers with source files as in:
/EXCALL /INC TEST.F90(3-41,50-67)
In this example, the /EXCALL option excludes all modules from the profile, and file://C:\TEMP\~hh10D5.htm 6/21/01
Page 65 of 79
the /INC option supercedes that to include only lines 3 - 41 and lines 50 - 67 from the source file TEST.F90. Note the absence of spaces in the source specification. To specify all source lines in a particular module, specify the .OBJ file like this:
/EXCALL /INC TEST.OBJ
or by using the source filename with zero line numbers like this:
/EXCALL /INC TEST.F90(0-0)
The following statement profiles from line 50 to the end of the file:
/EXCALL /INC TEST.F90(50-0)
PROFILE
PROFILE profiles an application and generates a .PBO file of the results. Use PROFILE after creating a .PBI file with PREP. The syntax for PROFILE is: PROFILE [options] programname [programargs] PROFILE reads the command line from left to right, so the rightmost options override contradictory options to the left. None of the options are case sensitive. You must prefix options with a forward slash (/) or a dash (-), and options must be separated by spaces. If you do not specify a .PBO filename on the command line, PROFILE uses the base name of the .PBI file with a .PBO extension. If you do not specify a .PBI or a .PBO file, PROFILE uses the base name of programname with the .PBI and .PBO extensions. Parameter options programname programargs Description Control .PBI input, .PBO output, error printing, and other profiler features. (See the PROFILE Options table). Filename of program to profile. PROFILE adds the .EXE extension if no extension is given. (See the Remarks section.) Optional command-line arguments for programname. (See the Remarks section.)
Page 66 of 79
Option /A
Description Appends any redirected error messages to an existing file. If the /E command-line option is used without the /A option, the file is overwritten. This option is valid only with the /E option. Sends profiler-generated error messages to filename. Provides a short summary of PROFILE options. Specifies a .PBI file to be read. This file is generated by PREP. Suppresses the PROFILE copyright message. Specifies a .PBO file to be generated. Use the PREP utility to merge with other .PBO files or to create a .PBT file for use with PLIST. Returns the exit code of the program being profiled. Provides a short summary of PROFILE options.
/X /?
Environment Variable The PROFILE environment variable specifies the default command-line options for PROFILE. If the PROFILE environment variable is not specified, there are no defaults. Remarks You must specify the filename of the program to profile on the PROFILE command line. PROFILE assumes the .EXE extension, if no extension is given. You can follow the program name with command-line arguments; these arguments are passed to the profiled program unchanged. If you are profiling code in a .DLL file, give the name of an executable file that calls it. For example, if you want to profile SAMPLE.DLL, which is called by CALLER.EXE, you can type:
PROFILE CALLER.EXE
PLIST
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 67 of 79
PLIST converts results from the .PBT file generated by the second call to PREP into a formatted text file. The syntax for PLIST is: PLIST [options] inputfile PLIST reads the command line from left to right, so the rightmost options override contradictory options to the left. None of the options are case sensitive. You must prefix options with a forward slash (/) or a dash (-), and options must be separated by spaces. PLIST results are sent to STDOUT by default. Use the greater-than (>) redirection PLIST must be run from the directory in which the profiled program was compiled. Parameter options inputfile Description Control the format and organization of profiler output data. (See the PLIST Options table.) The .PBT file to be converted by PLIST.
PLIST Options Option /C count /D directory Description Specifies the minimum hit count to appear in the listing. Specifies an additional directory for PLIST to search for source files. Use multiple /D command-line options to specify multiple directories. Use this option when PLIST cannot find a source file. Lists full paths in tab-delimited report. When using function attribution (see PREP /AT), displays function attribution with no indentation. Provides a short summary of PLIST options.
/F /FLAT /H[ELP]
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 68 of 79
When using function attribution (see PREP /AT), displays function attribution information in indented format. This is the default display for function attribution if neither /FLAT nor /TAB is selected. Suppresses the PLIST copyright message. Sets page length (in lines) of output. The length must be 0 or 15-255. A length of 0 suppresses page breaks. The default length is 0. Sets page width (in characters) of output. The width must be 1511. The default width is 511. Sorts output by counts, highest first. Sorts output in the order that the lines appear in the file. This is the default. This option is available only when profiling by line. Forces line count profile output to be printed in coverage format. Sorts output in alphabetical order by function name. This option is available only when profiling by function. Displays function timing or function counting information in function coverage format. Sorts output in alphabetical order by function name. Sorts output by time, highest first. Tab-separated output. Generates a tab-delimited database from the .PBT file for export to other applications. All other options, including sort specifications, are ignored when using this option. For more information, see Exporting Data from the Profiler. When using function attribution (see PREP /AT), sets tab width for indentation of function information. Provides a summary of PLIST options.
/ST /T
/TAB indent /?
Environment Variable The PLIST environment variable specifies the default command-line options for PLIST. If the PLIST environment variable is not specified, the default options for PLIST depend on the profile type as shown:
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 69 of 79
Profile Type Function timing Function counting Function coverage Line counting Line coverage
Tab-Delimited Record Format Global Information Records Local Information Records Steps to Analyze Profiler Statistics Processing Profiler Output with Microsoft Excel Generating the Tab-Delimite Report Using the PROFILER.XLM Macro Changing the PROFILER.XLM Selection Criteria
The PLIST /T command-line option causes PLIST to dump the contents of a .PBT file into a tab-delimited format suitable for import into a spreadsheet or database. This format can also be used by user-written programs. For example, to create a tab-delimited file called MYPROG.TXT from MYPROG.PBT, enter:
PLIST /T MYPROG > MYPROG.TXT
The ASCII tab-delimited format was designed to be read by other programs; it was not intended for general reporting.
Page 70 of 79
included functions will be listed for function counting and timing profiles, even if those functions were not executed. The tab-delimited format is arranged with one record per line and two to eight fields per record. The following figure shows how a database looks when loaded into Microsoft Excel. The database was produced using the PLIST /T commandline option.
The first item in each record is a format tag number. These tags range from 0 to 7 and indicate the kind of data given in the other fields of the record. The fields in each record are described in:
l l
Tab-delimited reports are generated with global information records first, organized in numerical order by format tag. The local information records, containing information about specific lines or functions, are generated last. Local information records are organized by line number. If the .PBT file contains information from more than one .EXE or .DLL file, the global information will cover them all. Local information records include the EXE field, which specifies the name of the executable file that each record pertains to.
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 71 of 79
Field 0
Profiling Method 1 Method Description Explanation Format tag number Numeric value that indicates the profiling type (see the following table)
Field 1 Method
Description ASCII description of the profiling type given by the method field The profiling types are listed in the following table: Profiling Types Method 321 324 521 522 524 Description Profile: Line counting, sorted by line Profile: Line coverage, sorted by line Profile: Function counting, sorted by function name Profile: Function timing, sorted by function name Profile: Function coverage, sorted by function name
Profiling Time and Depth 2 Total Time Outside Time Call Depth
file://C:\TEMP\~hh10D5.htm
6/21/01
Using Visual Fortran Tools Field 2 Total Time Outside Time Call Depth Explanation Format tag number
Page 72 of 79
Total amount of time used by the program being profiled. This field is zero for counting and coverage profiles Amount of time spent before the first profiled function (with function profiling) or line (with line profiling) was executed. This field is zero for counting and coverage profiles Maximum number of nested functions found while profiling. Only profiled functions are counted. This field is zero for line-level profiling
Explanation Format tag number Total number of times the profiler detected a profiled line or function being executed Total number of lines or functions marked for profiling Number of marked lines or functions executed at least once while profiling
Profiling Date/Command Line 4 Date Command Line Explanation Format tag number The date/time the profile was run (ASCII format)
Field 4 Date
Command Line The PLIST command-line arguments Profiling Starting Function Name 5 Starting Function Name
file://C:\TEMP\~hh10D5.htm
6/21/01
Using Visual Fortran Tools Field 5 Starting Function Name Explanation Format tag number
Page 73 of 79
The decorated name of the starting function identified by the PREP /SF parameter
Field 6 Exe
Explanation Format tag number ASCII name of the executable file that contains this function.
Source ASCII name of the object module (including the .OBJ extension) that contains this function Count Time Child Func Number of times this function has been executed Amount of time spent executing this function in milliseconds. This field is zero with profiling by counting or coverage Amount of time spent executing the function and any child functions it calls. This field is zero with profiling by counting or coverage ASCII name of the function
file://C:\TEMP\~hh10D5.htm
6/21/01
Using Visual Fortran Tools Field 7 Exe Explanation Format tag number
Page 74 of 79
Source ASCII name of the source that contains the first line of this function Line Count Line number of this line Number of times this line has been executed. With coverage, this field is 1 if the line has been executed and 0 otherwise
Profiling Function-Attribution Stacks 8 Field 8 Number of stacks Explanation Format tag number
Number of stacks Number of stacks for each function call Profiling Stack Hits and Timing 9 Stack size Hit count Stack time Child time
Explanation Format tag number Stack size for each function call Hit count for this stack
Stack time This stack's self-time Child time This stack's child-time
file://C:\TEMP\~hh10D5.htm
6/21/01
Using Visual Fortran Tools Field 10 Function name Explanation Format tag number
Page 75 of 79
Each function's name on the stack. The number of names that appear here will be equal to the stack size field in Profiling Stack Hits and Timing (above).
Page 76 of 79
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 77 of 79
l l
FSPLIT and FSPLIT90 The FSPLIT and FSPLIT90 tools split a multi-routine Fortran file into individual files. These tools are useful if you have a large Fortran program. Use FSPLIT90 when your program uses free-form source or Fortran 95/90 constructs. Use FSPLIT for FORTRAN 77 code. The FSPLIT and FSPLIT90 commands have the same form: FSPLIT [ options ] [input-file...] FSPLIT90 [ options ] [input-file...] options Is one of the keywords listed in the following table. If more than one option is specified, separate each with a space. The command options are: Option
-e:name
Description Processes only the program unit name. You can specify more than one -e name on a command line. Treats the statement field of each source line as ending in column 132, instead of column 72. Displays information about the FSPLIT command. Suppresses the copyright notice that is displayed when FSPLIT or FSPLIT90 is run. Suppresses display of the name of each file opened (input and output files). A Fortran source file to be split. You can specify more than one file by using a list of files. If input-file is omitted the FSPLIT or FSPLIT90 Utility reads from standard input.
-silent
input-file
input-file A Fortran source file to be split. You can specify more than one file by using a list of files. If input-file is omitted the FSPLIT or FSPLIT90 Utility reads from standard input.
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 78 of 79
FSPLIT or FSPLIT90 splits multi-routine Fortran files into separate routine files of the form filename.for, where filename is the name of the program unit (for example: a function, subroutine, block data, or program). The name for unnamed block data subprograms has the form blkdtannn.for, where nnn is a 3digit code. For unnamed main programs, the name has the form mainnnn.for. If there is an error in classifying a program unit, or if filename.for already exists, the program unit is put in a file named zzznnn.for, where nnn is a 3-digit code. Normally each subprogram unit is split into a separate file. Avoid using the -e option for unnamed main programs and block data subprograms since you must predict the created file name. If FSPLIT or FSPLIT90 cannot find the names specified by the -e option, an error message is written to standard error device. The following command example splits the subprogram units readit and doit into separate files:
FSPLIT -e readit -e doit prog.for
FPR The FPR tool transforms files formatted according to Fortran's carriage control conventions into files formatted according to line printer conventions. The FPR command has the following form: FPR [-f record-size] [filename] record-size Specifies a fixed-length record as input. The record-size must be a decimal integer. filename Specifies the data file to be transformed. FPR copies the input filename onto itself, replacing the carriage control characters with characters that will produce the intended effects when printed using the PRINT command. The first character of each line determines the vertical spacing as follows:
file://C:\TEMP\~hh10D5.htm
6/21/01
Page 79 of 79
Vertical Space Before Printing One line Two lines To first line of next page No advance One line; no return after printing
FPR interprets the first character of every line of input, even if that character is not a recognizable control character. Control characters that are not recognized are treated as blanks and result in a single line advance. FPR handles stream and fixed-length files. Input to FPR is assumed to be a stream (Stream_LF) file, unless you specify the -f option. No diagnostic message is issued when FPR encounters an unrecognized control character.
file://C:\TEMP\~hh10D5.htm
6/21/01