Skip to content
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

Optimzations make assumption about function names #130172

Closed
smaillet opened this issue Mar 6, 2025 · 3 comments
Closed

Optimzations make assumption about function names #130172

smaillet opened this issue Mar 6, 2025 · 3 comments
Labels
llvm:optimizations question A question, not bug report. Check out https://llvm.org/docs/GettingInvolved.html instead!

Comments

@smaillet
Copy link

smaillet commented Mar 6, 2025

Given the following module as an input

define double @anon_expr_1() {
entry:
  %calltmp = call double @cos(double 1.234000e+00)
  ret double %calltmp
}

declare double @cos(double)

optimization with a minimal set of passes "simplifycfg,instcombine,reassociate,gvn" will produce the following module:

define double @anon_expr_1() {
entry:
  %calltmp = call double @cos(double 1.234000e+00)
  ret double 0x3FD526571FE8C7A5
}

declare double @cos(double)

NOTE: The return of this function is the IEEE Hex form of 0.33046510807172985 which is the mathematical Cosine of 1.234. Thus, the optimizations have Assumed that the function named cos is in fact the mathematical function of the same name common in C runtime libraries - but it doesn't KNOW that. It could well be that this is just a function that happens to have the name cos (and has NO reliance on the C runtime). The expectation is that this does something completely different that might even have side effects so ignoring the return value and substituting a const is just WRONG behavior. (One would hope that subsequent optimization stages would not remove the call due to the possibility of side effects, but ignoring the return and substituting a const value is still wrong)

@nikic
Copy link
Contributor

nikic commented Mar 6, 2025

You can place the nobuiltin attribute on the call to disable libcall recognition.

Like this:

define double @anon_expr_1()  {
entry:
  %calltmp = call double @cos(double 1.234000e+00) nobuiltin
  ret double %calltmp
}

declare double @cos(double)

@nikic nikic closed this as not planned Won't fix, can't repro, duplicate, stale Mar 6, 2025
@nikic nikic added the question A question, not bug report. Check out https://llvm.org/docs/GettingInvolved.html instead! label Mar 6, 2025
@nikic
Copy link
Contributor

nikic commented Mar 6, 2025

Alternatively, you can also use TargetLibraryInfo to configure which libcalls are available in your environment. disableAllFunctions() disables all libcalls.

@smaillet
Copy link
Author

smaillet commented Mar 6, 2025

requiring the "nobuiltin" seems like a dangerous form of OPT-IN to me. (Note, I found this when converting OLD code from LLVM 10 to an RC of LLVM 20 so it wasn't always like this). Use of the TargetLibraryInfo is something to consider, except that this is NOT currently using any target library. At most such a thing is tied to the TargetMachine, which is currently set to "NULL" for the optimization runs as there isn't any particular target yet. (Just IR). I am also constrained by the LLVM-C API so, no access to the underlying C++. Sadly, the LLVM-C API has no mention of TargetLibraryInfo. So, to use that I'd need some sort of custom extension wrapper.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
llvm:optimizations question A question, not bug report. Check out https://llvm.org/docs/GettingInvolved.html instead!
Projects
None yet
Development

No branches or pull requests

4 participants