Skip to content

[libc] sqrt and log functions fuzz tests #148006

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 6 commits into from
Jul 15, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
54 changes: 45 additions & 9 deletions libc/fuzzing/math/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -98,15 +98,6 @@ add_libc_fuzzer(
libc.src.math.cos
)

add_libc_fuzzer(
atan_fuzz
NEED_MPFR
SRCS
atan_fuzz.cpp
DEPENDS
libc.src.math.atan
)

add_libc_fuzzer(
tan_fuzz
NEED_MPFR
Expand All @@ -124,3 +115,48 @@ add_libc_fuzzer(
DEPENDS
libc.src.math.sincos
)

add_libc_fuzzer(
log_fuzz
NEED_MPFR
SRCS
log_fuzz.cpp
DEPENDS
libc.src.math.log
)

add_libc_fuzzer(
log10_fuzz
NEED_MPFR
SRCS
log10_fuzz.cpp
DEPENDS
libc.src.math.log10
)

add_libc_fuzzer(
log1p_fuzz
NEED_MPFR
SRCS
log1p_fuzz.cpp
DEPENDS
libc.src.math.log1p
)

add_libc_fuzzer(
log2_fuzz
NEED_MPFR
SRCS
log2_fuzz.cpp
DEPENDS
libc.src.math.log2
)

add_libc_fuzzer(
sqrt_fuzz
NEED_MPFR
SRCS
sqrt_fuzz.cpp
DEPENDS
libc.src.__support.FPUtil.generic.sqrt
)
38 changes: 0 additions & 38 deletions libc/fuzzing/math/atan_fuzz.cpp

This file was deleted.

51 changes: 51 additions & 0 deletions libc/fuzzing/math/log10_fuzz.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
//===-- log10_fuzz.cpp ----------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
///
/// Fuzzing test for llvm-libc log10 implementation.
///
//===----------------------------------------------------------------------===//

#include "src/math/log10.h"
#include "utils/MPFRWrapper/mpfr_inc.h"
#include <cstdint>
#include <cstring>
#include <iostream>
#include <math.h>

extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
mpfr_t input;
mpfr_init2(input, 53);
for (size_t i = 0; i < size / sizeof(double); ++i) {
double x;
std::memcpy(&x, data, sizeof(double));
data += sizeof(double);

// remove NaN and inf and values outside accepted range
if (isnan(x) || isinf(x) || x < 0)
return 0;
// signed zeros already tested in unit tests
if (signbit(x) && x == 0.0)
return 0;

mpfr_set_d(input, x, MPFR_RNDN);
int output = mpfr_log10(input, input, MPFR_RNDN);
mpfr_subnormalize(input, output, MPFR_RNDN);
double to_compare = mpfr_get_d(input, MPFR_RNDN);

double result = LIBC_NAMESPACE::log10(x);

if (result != to_compare) {
std::cout << std::hexfloat << "Failing input: " << x << std::endl;
std::cout << std::hexfloat << "Failing output: " << result << std::endl;
std::cout << std::hexfloat << "Expected: " << to_compare << std::endl;
__builtin_trap();
}
}
mpfr_clear(input);
return 0;
}
50 changes: 50 additions & 0 deletions libc/fuzzing/math/log1p_fuzz.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
//===-- log1p_fuzz.cpp ----------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
///
/// Fuzzing test for llvm-libc log1p implementation.
///
//===----------------------------------------------------------------------===//

#include "src/math/log1p.h"
#include "utils/MPFRWrapper/mpfr_inc.h"
#include <cstdint>
#include <cstring>
#include <iostream>
#include <math.h>

extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
mpfr_t input;
mpfr_init2(input, 53);
for (size_t i = 0; i < size / sizeof(double); ++i) {
double x;
std::memcpy(&x, data, sizeof(double));
data += sizeof(double);
// remove NaN and inf and values outside accepted range
if (isnan(x) || isinf(x) || x < -1)
return 0;
// signed zeros already tested in unit tests
if (signbit(x) && x == 0.0)
return 0;

mpfr_set_d(input, x, MPFR_RNDN);
int output = mpfr_log1p(input, input, MPFR_RNDN);
mpfr_subnormalize(input, output, MPFR_RNDN);
double to_compare = mpfr_get_d(input, MPFR_RNDN);

double result = LIBC_NAMESPACE::log1p(x);

if (result != to_compare) {
std::cout << std::hexfloat << "Failing input: " << x << std::endl;
std::cout << std::hexfloat << "Failing output: " << result << std::endl;
std::cout << std::hexfloat << "Expected: " << to_compare << std::endl;
__builtin_trap();
}
}
mpfr_clear(input);
return 0;
}
51 changes: 51 additions & 0 deletions libc/fuzzing/math/log2_fuzz.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
//===-- log2_fuzz.cpp -----------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
///
/// Fuzzing test for llvm-libc log2 implementation.
///
//===----------------------------------------------------------------------===//

#include "src/math/log2.h"
#include "utils/MPFRWrapper/mpfr_inc.h"
#include <cstdint>
#include <cstring>
#include <iostream>
#include <math.h>

extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
mpfr_t input;
mpfr_init2(input, 53);
for (size_t i = 0; i < size / sizeof(double); ++i) {
double x;
std::memcpy(&x, data, sizeof(double));
data += sizeof(double);

// remove NaN and inf and values outside accepted range
if (isnan(x) || isinf(x) || x < 0)
return 0;
// signed zeros already tested in unit tests
if (signbit(x) && x == 0.0)
return 0;

mpfr_set_d(input, x, MPFR_RNDN);
int output = mpfr_log2(input, input, MPFR_RNDN);
mpfr_subnormalize(input, output, MPFR_RNDN);
double to_compare = mpfr_get_d(input, MPFR_RNDN);

double result = LIBC_NAMESPACE::log2(x);

if (result != to_compare) {
std::cout << std::hexfloat << "Failing input: " << x << std::endl;
std::cout << std::hexfloat << "Failing output: " << result << std::endl;
std::cout << std::hexfloat << "Expected: " << to_compare << std::endl;
__builtin_trap();
}
}
mpfr_clear(input);
return 0;
}
50 changes: 50 additions & 0 deletions libc/fuzzing/math/log_fuzz.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
//===-- log_fuzz.cpp ------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
///
/// Fuzzing test for llvm-libc log implementation.
///
//===----------------------------------------------------------------------===//

#include "src/math/log.h"
#include "utils/MPFRWrapper/mpfr_inc.h"
#include <cstdint>
#include <cstring>
#include <iostream>
#include <math.h>

extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
mpfr_t input;
mpfr_init2(input, 53);
for (size_t i = 0; i < size / sizeof(double); ++i) {
double x;
std::memcpy(&x, data, sizeof(double));
data += sizeof(double);

// remove NaN and inf and values outside accepted range
if (isnan(x) || isinf(x) || x < 0)
return 0;
// signed zeros already tested in unit tests
if (signbit(x) && x == 0.0)
return 0;
mpfr_set_d(input, x, MPFR_RNDN);
int output = mpfr_log(input, input, MPFR_RNDN);
mpfr_subnormalize(input, output, MPFR_RNDN);
double to_compare = mpfr_get_d(input, MPFR_RNDN);

double result = LIBC_NAMESPACE::log(x);

if (result != to_compare) {
std::cout << std::hexfloat << "Failing input: " << x << std::endl;
std::cout << std::hexfloat << "Failing output: " << result << std::endl;
std::cout << std::hexfloat << "Expected: " << to_compare << std::endl;
__builtin_trap();
}
}
mpfr_clear(input);
return 0;
}
2 changes: 1 addition & 1 deletion libc/fuzzing/math/sincos_fuzz.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
//
//===----------------------------------------------------------------------===//
///
/// Fuzzing test for llvm-libc cos implementation.
/// Fuzzing test for llvm-libc sincos implementation.
///
//===----------------------------------------------------------------------===//

Expand Down
50 changes: 50 additions & 0 deletions libc/fuzzing/math/sqrt_fuzz.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
//===-- sqrt_fuzz.cpp -----------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
///
/// Fuzzing test for llvm-libc sqrt implementation.
///
//===----------------------------------------------------------------------===//

#include "src/__support/FPUtil/generic/sqrt.h"
#include "utils/MPFRWrapper/mpfr_inc.h"
#include <cstdint>
#include <cstring>
#include <iostream>
#include <math.h>

extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
mpfr_t input;
mpfr_init2(input, 53);
for (size_t i = 0; i < size / sizeof(double); ++i) {
double x;
std::memcpy(&x, data, sizeof(double));
data += sizeof(double);
// remove NaN and inf and values outside accepted range
if (isnan(x) || isinf(x) || x < 0)
return 0;
// signed zeros already tested in unit tests
if (signbit(x) && x == 0.0)
return 0;

mpfr_set_d(input, x, MPFR_RNDN);
int output = mpfr_sqrt(input, input, MPFR_RNDN);
mpfr_subnormalize(input, output, MPFR_RNDN);
double to_compare = mpfr_get_d(input, MPFR_RNDN);

double result = LIBC_NAMESPACE::fputil::sqrt<double>(x);

if (result != to_compare) {
std::cout << std::hexfloat << "Failing input: " << x << std::endl;
std::cout << std::hexfloat << "Failing output: " << result << std::endl;
std::cout << std::hexfloat << "Expected: " << to_compare << std::endl;
__builtin_trap();
}
}
mpfr_clear(input);
return 0;
}
Loading