DOCUMENT:Q63360 04-MAY-2001 [fortran] TITLE :FIX: INTEGER*2 Multiply by Power of 2 Produces Wrong Results PRODUCT :Microsoft Fortran Compiler PROD/VER::5.0 OPER/SYS: KEYWORDS: ====================================================================== ------------------------------------------------------------------------------- The information in this article applies to: - Microsoft FORTRAN compiler for OS/2, version 5.0 - Microsoft FORTRAN Compiler for MS-DOS, version 5.0 ------------------------------------------------------------------------------- SYMPTOMS ======== An application that multiplies an INTEGER*2 variable by an integer constant that is a power of 2 produces incorrect results. Specifying the /4I4 compiler option switch does not have any effect on the results. Compiling the code with the /4Yb option and running the application does not provide any indication of a problem. Compiling the code with the /Od option to disable optimizations and running the application does not change the behavior. CAUSE ===== The compiler generates incorrect code when the result of the multiplication is larger than the range of an INTEGER*2 variable (-32,768 to 32,767). RESOLUTION ========== To address this problem, perform one of the following steps: - Modify the source code to use the INT4 intrinsic function to convert the INTEGER*2 variable to an INTEGER*4 variable in the expression. - Modify the source code to assign the INTEGER*2 variable to a temporary INTEGER*4 variable. Use the temporary variable in the expression. - Modify the source code to assign the integer constant to a temporary variable. Use the temporary variable in the expression. STATUS ====== Microsoft has confirmed this to be a problem in FORTRAN version 5.0 for MS-DOS and OS/2. This problem was corrected in FORTRAN version 5.1 for MS-DOS and OS/2. MORE INFORMATION ================ This error occurs when the result of the multiplication is larger than the range of an INTEGER*2 variable (-32,768 to 32,767). The compiler generates a shift instruction (SHL) to perform a fast multiply. The compiler does not use a four byte work space when it performs the shift; instead, it allocates only two bytes for its work space. Under these circumstances, significant bits can be lost or shifted into the sign bit when magnitude of the result of the multiplication is too large for an INTEGER*2 variable. The following code example demonstrates this problem and one technique to work around it. Sample Code ----------- C Compile options needed: None INTEGER*2 K/1234/ INTEGER*2 M/32/, N/33/ WRITE(*, *) K*32, K*33 ! Displays incorrect results for K*32 WRITE(*, *) K*M, K*N ! Displays correct results for K*M END Additional query words: 5.00 buglist5.00 fixlist5.10 ====================================================================== Keywords : Technology : kbAudDeveloper kbFortranSearch kbZNotKeyword3 kbFORTRAN500DOS kbFORTRAN500OS2 Version : :5.0 Issue type : kbbug Solution Type : kbfix ============================================================================= THE INFORMATION PROVIDED IN THE MICROSOFT KNOWLEDGE BASE IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND. MICROSOFT DISCLAIMS ALL WARRANTIES, EITHER EXPRESS OR IMPLIED, INCLUDING THE WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT SHALL MICROSOFT CORPORATION OR ITS SUPPLIERS BE LIABLE FOR ANY DAMAGES WHATSOEVER INCLUDING DIRECT, INDIRECT, INCIDENTAL, CONSEQUENTIAL, LOSS OF BUSINESS PROFITS OR SPECIAL DAMAGES, EVEN IF MICROSOFT CORPORATION OR ITS SUPPLIERS HAVE BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. SOME STATES DO NOT ALLOW THE EXCLUSION OR LIMITATION OF LIABILITY FOR CONSEQUENTIAL OR INCIDENTAL DAMAGES SO THE FOREGOING LIMITATION MAY NOT APPLY. Copyright Microsoft Corporation 2001.