Subversion Repositories shark

Rev

Rev 1619 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | RSS feed

/* Project:     OSLib
 * Description: The OS Construction Kit
 * Date:                1.6.2000
 * Idea by:             Luca Abeni & Gerardo Lamastra
 *
 * OSLib is an SO project aimed at developing a common, easy-to-use
 * low-level infrastructure for developing OS kernels and Embedded
 * Applications; it partially derives from the HARTIK project but it
 * currently is independently developed.
 *
 * OSLib is distributed under GPL License, and some of its code has
 * been derived from the Linux kernel source; also some important
 * ideas come from studying the DJGPP go32 extender.
 *
 * We acknowledge the Linux Community, Free Software Foundation,
 * D.J. Delorie and all the other developers who believe in the
 * freedom of software and ideas.
 *
 * For legalese, check out the included GPL license.
 */


#include <arch/i386/stdlib.h>
#include <arch/i386/string.h>
#include <arch/i386/limits.h>
#include <arch/i386/float.h>
#include <ll/i386/mem.h>
#include <arch/stdarg.h>
#include <arch/ctype.h>
#include <arch/math.h>
#include "sprintf.h"

FILE(fcvt);

unsigned fcvt(double v,char *buffer,int width,int prec,int flag)
{
    double ip,fp,val;
    register int l=0;
    register int digits;
    int conv,got_a_digit,exceeded,len = 0;
    char tmp[300];
    memset(tmp,0,300);
    /* Manage Inf & NaN */
    if (isspecial(v,buffer)) return(strlen(buffer));
    /* Check for negative value & add optional + sign */
    if (v < 0) {
        *buffer++ = '-';
        v = -v;
        len++;
    } else if (flag & ADD_PLUS) {
        *buffer++ = '+';
        len++;
    }
    /* Extract integer part & mantissa */
    fp = modf(v,&ip);
    /* Process integer part */
    digits = 0;
    if (ip >= 1.0) {
        while (ip >= 1.0) {
            /* The 0.01 here is used to correct rounding errors*/
            /* Ex: 1.2 --> 1.1999999999999... adjust to 1.2 */
            /*
            ip = ip / 10.0;
            modf(ip,&val);
            val = (ip - val)*10.0;
            */

            val = fmod(ip,10.0);
            ip = ip / 10.0;
            tmp[digits++] = todigit((unsigned)(val));
        }
        len += digits;
        /* Now reverse the temporary buffer into output buffer  */
        /* Translate only the last 15 digits                    */
        /* The others are beyond double precision limit!        */     
        for (l = digits-1; l >= max(digits-15,0); l--) *buffer++ = tmp[l];
        if (l >= 0) for (l = digits-16; l >= 0; l--) *buffer++ = '0';
        *buffer = 0;
    /* If IP == 0 -> just put in a 0 */
    } else {
        *buffer++ = '0';
        len++;
    }
    /* Process fractionary part according to width specification */
    /* If RESPECT_WIDTH is set, scan until you reach wanted precision */
    /* Else scan until you find a not 0 digit */
    if (fp > 1e-307 && len < width+1) {
        *buffer++ = '.';
        len++;
        if (flag & RESPECT_WIDTH) got_a_digit = 1;
        else got_a_digit = 0;
        exceeded = 0;
        digits = 1;
        while (!exceeded) {
            /* The DBL_EPSILON here is used to correct rounding errors */
            fp = (fp + DBL_EPSILON) * 10.0;
            fp = modf(fp,&ip);
            conv = (int)(ip);
            if (conv != 0 && !got_a_digit) got_a_digit = 1;
            *buffer++ = todigit(conv);
            len++;
            digits++;
            if (got_a_digit && (digits > prec)) exceeded = 1;
            if (width < len) exceeded = 1;
        }
        /* Drop trailing zeros after decimal point */
        while (*--buffer == '0' && *(buffer-1) != '.') len--;
        buffer++;
    }
    *buffer = 0;
    return(len);
}