/*
 * strutils.c - various string routines shared by commands
 * This file was copied from util-linux at fall 2011.
 *
 * Copyright (C) 2010 Karel Zak <kzak@redhat.com>
 * Copyright (C) 2010 Davidlohr Bueso <dave@gnu.org>
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
 */

#include <stdlib.h>
#include <ctype.h>

#include "c.h"
#include "strutils.h"

/*
 * same as strtol(3) but exit on failure instead of returning crap
 */
long strtol_or_err(const char *str, const char *errmesg)
{
	long num;
	char *end = NULL;

	if (str != NULL && *str != '\0') {
		errno = 0;
		num = strtol(str, &end, 10);
		if (errno == 0 && str != end && end != NULL && *end == '\0')
			return num;
	}
	error(EXIT_FAILURE, errno, "%s: '%s'", errmesg, str);
	return 0;
}

/*
 * same as strtod(3) but exit on failure instead of returning crap
 */
double strtod_or_err(const char *str, const char *errmesg)
{
	double num;
	char *end = NULL;

	if (str != NULL && *str != '\0') {
		errno = 0;
		num = strtod(str, &end);
		if (errno == 0 && str != end && end != NULL && *end == '\0')
			return num;
	}
	error(EXIT_FAILURE, errno, "%s: '%s'", errmesg, str);
	return 0;
}

/*
 * Covert a string into a double in a non-locale aware way.
 * This means the decimal point can be either . or ,
 * Also means you cannot use the other for thousands separator
 *
 * Exits on failure like its other _or_err cousins
 */
double strtod_nol_or_err(char *str, const char *errmesg)
{
    double num;
    const char *cp, *radix;
    double mult;
    int negative = 0;

    if (str != NULL && *str != '\0') {
        num = 0.0;
        cp = str;
        /* strip leading spaces */
        while (isspace(*cp))
            cp++;

        /* get sign */
        if (*cp == '-') {
            negative = 1;
            cp++;
        } else if (*cp == '+')
            cp++;

        /* find radix */
        radix = cp;
        mult=0.1;
        while(isdigit(*radix)) {
            radix++;
            mult *= 10;
        }
        while(isdigit(*cp)) {
            num += (*cp - '0') * mult;
            mult /= 10;
            cp++;
        }
        /* got the integers */
        if (*cp == '\0')
            return (negative?-num:num);
        if (*cp != '.' && *cp != ',')
            error(EXIT_FAILURE, EINVAL, "%s: '%s'", errmesg, str);

        cp++;
        mult = 0.1;
        while(isdigit(*cp)) {
            num += (*cp - '0') * mult;
            mult /= 10;
            cp++;
        }
        if (*cp == '\0')
            return (negative?-num:num);
    }
    error(EXIT_FAILURE, errno, "%s: '%s'", errmesg, str);
    return 0;
}