usr/src/lib/libc/port/gen/strtod.c
author raf
Mon, 10 Apr 2006 12:27:38 -0700
changeset 1778 6357a59054f7
parent 0 68f95e015346
child 6812 febeba71273d
permissions -rw-r--r--
6404383 select() behaviour changed in Solaris 10, breaking binary compatibility
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     1
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     2
 * CDDL HEADER START
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     3
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     4
 * The contents of this file are subject to the terms of the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     5
 * Common Development and Distribution License, Version 1.0 only
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     6
 * (the "License").  You may not use this file except in compliance
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     7
 * with the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     8
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     9
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    10
 * or http://www.opensolaris.org/os/licensing.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    11
 * See the License for the specific language governing permissions
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    12
 * and limitations under the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    13
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    14
 * When distributing Covered Code, include this CDDL HEADER in each
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    15
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    16
 * If applicable, add the following below this CDDL HEADER, with the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    17
 * fields enclosed by brackets "[]" replaced with your own identifying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    18
 * information: Portions Copyright [yyyy] [name of copyright owner]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    19
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    20
 * CDDL HEADER END
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    21
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    22
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    23
 * Copyright 2004 Sun Microsystems, Inc.  All rights reserved.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    24
 * Use is subject to license terms.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    25
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    26
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    27
#pragma ident	"%Z%%M%	%I%	%E% SMI"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    29
/*	Copyright (c) 1988 AT&T	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    30
/*	  All Rights Reserved  	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    32
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
#include "synonyms.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
#include <errno.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    35
#include <stdio.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
#include <values.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
#include <floatingpoint.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
#include <stdlib.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
#include <sys/types.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
#include "libc.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
#include "xpg6.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
double
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
strtod(const char *cp, char **ptr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
	double		x;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
	decimal_mode	mr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
	decimal_record  dr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    49
	fp_exception_field_type fs;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
	enum decimal_string_form form;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    51
	char		*pechar;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
	int		lc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    53
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    54
	lc = (__xpg6 & _C99SUSv3_recognize_hexfp)? -1 : 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    55
	string_to_decimal((char **)&cp, MAXINT, lc, &dr, &form, &pechar);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    56
	if (ptr != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    57
		*ptr = (char *)cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
	if (form == invalid_form)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
		return (0.0);	/* Shameful kluge for SVID's sake. */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
#if defined(__sparc)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    61
	mr.rd = _QgetRD();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
#elif defined(__i386) || defined(__amd64)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
	mr.rd = __xgetRD();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    64
#else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
#error Unknown architecture
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
	if ((int)form < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    68
		__hex_to_double(&dr, mr.rd, &x, &fs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    69
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
		decimal_to_double(&x, &mr, &dr, &fs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
	if (fs & ((1 << fp_overflow) | (1 << fp_underflow)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
		errno = ERANGE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
	return (x);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    75
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    76
float
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    77
strtof(const char *cp, char **ptr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    78
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    79
	float		x;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    80
	decimal_mode	mr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
	decimal_record	dr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
	fp_exception_field_type fs;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    83
	enum decimal_string_form form;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
	char		*pechar;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
	string_to_decimal((char **)&cp, MAXINT, -1, &dr, &form, &pechar);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
	if (ptr != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
		*ptr = (char *)cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
	if (form == invalid_form)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
		return (0.0f);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    91
#if defined(__sparc)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    92
	mr.rd = _QgetRD();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
#elif defined(__i386) || defined(__amd64)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    94
	mr.rd = __xgetRD();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    95
#else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
#error Unknown architecture
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
	if ((int)form < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    99
		__hex_to_single(&dr, mr.rd, &x, &fs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
		decimal_to_single(&x, &mr, &dr, &fs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   102
	if (fs & ((1 << fp_overflow) | (1 << fp_underflow)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
		errno = ERANGE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   104
	return (x);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   106
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   107
long double
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   108
strtold(const char *cp, char **ptr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   109
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   110
	long double	x;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   111
	decimal_mode	mr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   112
	decimal_record	dr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   113
	fp_exception_field_type fs;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   114
	enum decimal_string_form form;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   115
	char		*pechar;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   116
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   117
	string_to_decimal((char **)&cp, MAXINT, -1, &dr, &form, &pechar);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   118
	if (ptr != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   119
		*ptr = (char *)cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   120
	if (form == invalid_form)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   121
		return (0.0L);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   122
#if defined(__sparc)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   123
	mr.rd = _QgetRD();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   124
	if ((int)form < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   125
		__hex_to_quadruple(&dr, mr.rd, &x, &fs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   126
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   127
		decimal_to_quadruple(&x, &mr, &dr, &fs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   128
#elif defined(__i386) || defined(__amd64)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   129
	mr.rd = __xgetRD();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   130
	if ((int)form < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   131
		__hex_to_extended(&dr, mr.rd, (extended *)&x, &fs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   132
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   133
		decimal_to_extended((extended *)&x, &mr, &dr, &fs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   134
#else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   135
#error Unknown architecture
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   136
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   137
	if (fs & ((1 << fp_overflow) | (1 << fp_underflow)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   138
		errno = ERANGE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   139
	return (x);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   140
}