log(10.0) can compile but log(0.0) cannot with undefined reference?

前端 未结 2 2079
闹比i
闹比i 2020-12-01 13:49

For the following C source code:

#include 

int main(void)
{
    double          x;

    x = log(0.0);

    return 0;
}
相关标签:
2条回答
  • 2020-12-01 14:08

    The compilation is alright, it's just the linker switch -lm that is missing.

    The second version probably compiles and links because gcc replaces log(10.0) with a constant, so no call to the math library is needed. In the second case, the result is mathematically undefined, and evaluation results in a domain error. In that case, the expression cannot be replaced by a constant, since handling of domain errors might be different at run-time.

    Quote from the C-standard (draft):

    On a domain error, the function returns an implementation-defined value; if the integer expression math_errhandling & MATH_ERRNO is nonzero, the integer expression errno acquires the value EDOM; if the integer expression math_errhandling & MATH_ERREXCEPT is nonzero, the ‘‘invalid’’ floating-point exception is raised.

    So evaluation of log(0.0) either results in returning the value HUGE_VAL (not NAN as I claimed before) or a floating point exception.

    EDIT: I corrected my answer based on the comments received and added link to the description in the C-standard.

    0 讨论(0)
  • 2020-12-01 14:10

    gcc can use builtin functions in many cases, their documentation says:

    Many of these functions are only optimized in certain cases; if they are not optimized in a particular case, a call to the library function is emitted.

    so therefore gcc will not need to link against the math library when using the builtin function but since log(0) is not defined it probably forcesgcc to evaluate it at run-time since it has a side effect.

    If we look at the draft C99 standard section 7.12.1 Treatment of error conditions in paragraph 4 it says (emphasis mine):

    A floating result overflows if the magnitude of the mathematical result is finite but so large that the mathematical result cannot be represented without extraordinary roundoff error in an object of the specified type. If a floating result overflows and default rounding is in effect, or if the mathematical result is an exact infinity from finite arguments (for example log(0.0)), then the function returns the value of the macro HUGE_VAL, HUGE_VALF, or HUGE_VALL according to the return type, with the same sign as the correct value of the function; if the integer expression math_errhandling & MATH_ERRNO is nonzero, the integer expression errno acquires the value ERANGE; if the integer expression math_errhandling & MATH_ERREXCEPT is nonzero, the ‘‘divide-by-zero’’ floating-point exception is raised if the mathematical result is an exact infinity and the ‘‘overflow’’ floating-point exception is raised otherwise.

    We can see from a live example using -S flag to generate assembly and grep log to filter out calls to log.

    In the case of log(0.0) the following instruction is generated (see it live):

    call    log
    

    but in the case of log(10.0) no call log instruction is generated, (see it live).

    We can usually prevent gcc from using builtin function by using the -fno-builtin flag which is probably a quicker way to test whether a builtin is being used.

    Note that -lm needs to go after the source file, for example (taken from linked answer) if main.c required the math library then you would use:

     gcc main.c -lm 
    
    0 讨论(0)
提交回复
热议问题