Skip to main content

compile - CompiledFunctionCall vs. LibraryFunction


When compiling code to C, and the compiled code calls other compiled functions, I see CompiledFunctionCall in the output of CompilePrint most of the time. But sometimes, it's replaced by LibraryFunction[<>,...]. Could anybody explain the difference between these two?


EDIT: As an example, I would expect a LibraryFunction call in the following case. If I define:


compBinomial=Compile[{{n,_Real},{k,_Real}},
Which[
k==0.||k==n,

1,
n==0.,
0,
k<=Quotient[n,2],
Product[(n+1-i)/i,{i,1,k}],
True,
Product[(n+1-i)/i,{i,1,n-k}]
]
,CompilationTarget->"C",RuntimeOptions->"Speed"
];


If I then define the following:


compsum=Compile[
{{n,_Integer},{m,_Integer},{j,_Integer}},
Round@Sum[
(-1)^k*compBinomial[n+m,n-k]*compBinomial[k+j,k]
,{k,0,n}]
,CompilationTarget->"C",RuntimeOptions->"Speed",
CompilationOptions->{"InlineExternalDefinitions" -> True,
"InlineCompiledFunctions"->False}

];

I see CompiledFunctionCall in the output of CompilePrint[compsum].



Answer



It appears that LibraryFunction occurs if the function being called inside the other compiled function is a function that has been compiled to C, provided this function has been called with the right type of arguments. It seems that in other cases there is a CompiledFunctionCall.


As pointed out by Simon Woods in the comments below, there is a type mismatch in the example that you added later, so that in this case CompiledFunctionCall is used.


Be warned that this is based on very few examples, so this may not be the correct explanation.


Here is a simple example where compiled functions are called with the right type of arguments. First we compile it to the WVM and then to C and observe the difference.


<< CompiledFunctionTools`
cf = Compile[{{x}}, Sin[x]];

cf5 = Compile[{{x}}, cf[x^2],
CompilationOptions -> {"InlineExternalDefinitions" -> True,
"InlineCompiledFunctions" -> False}];
cf5 // CompilePrint


...
R2 = CompiledFunctionCall[ Hold[CompiledFunction[{x}, Sin[x], \
-CompiledCode-]][ R1]]
...


cf2 = Compile[{{x}}, Sin[x], CompilationTarget -> "C"];
cf6 =
Compile[{{x}}, cf2[x^2],
CompilationOptions ->
{"InlineExternalDefinitions" -> True,
"InlineCompiledFunctions" -> False}
];
cf6 // CompilePrint



...
R2 = LibraryFunction[<>, compiledFunction12, {{Real, 0, Constant}}, \
Real][ R1]]
...

About LibraryFunction


In a compiled function that has been compiled to C, a LibraryFunction will do all the work. A library function is a function that is part of a dynamically linked library that is loaded into the kernel. The dynamically linked library is the result of compiling C code.


The following code creates an expression with head LibraryFunction. It does this by loading a pre-compiled function from a library.


fun = LibraryFunctionLoad["demo", "demo_I_I", {Integer}, Integer];

fun//Head


LibraryFunction

You can also see a LibraryFunction in the definition of cf2


cf2//InputForm


CompiledFunction[{10, 10.1, 

5468}, {_Real}, {{3, 0, 0}, {3, 0, 1}}, {}, {0, 0, 2, 0, 0},
{{40, 1, 3, 0, 0, 3, 0, 1}, {1}}, Function[{x}, Sin[x]], Evaluate,
LibraryFunction["path/compiledFunction12.dylib",
"compiledFunction12",
{{Real, 0, "Constant"}}, Real]]

In the definition of cf, no such LibraryFunction is present.


Skipping interaction with the CompiledFunction


The compilation process in Mathematica is smart enough to figure out that a compiled function is compiled to C, so it knows that it can simply call the LibraryFunction, rather than interacting with some CompiledFunction object. In fact we can do the same manually. Let


libFunc = cf2[[9]]



LibraryFunction[Function name: compiledFunction12 Argument count: 1]

So that we have defined libFunc as the LibraryFunction that does all the work for cf2.


Then we can do


libFunc[Pi]


 1.22465*10^-16


Conclusion


My guess is that this explains the difference. When calling a function that has been compiled to the WVM, a CompiledFunction is really called. When using a function that has been compiled to C, in the absence of a type mismatch,a shortcut is taken and the LibraryFunction is called directly.


Comments

Popular posts from this blog

plotting - Filling between two spheres in SphericalPlot3D

Manipulate[ SphericalPlot3D[{1, 2 - n}, {θ, 0, Pi}, {ϕ, 0, 1.5 Pi}, Mesh -> None, PlotPoints -> 15, PlotRange -> {-2.2, 2.2}], {n, 0, 1}] I cant' seem to be able to make a filling between two spheres. I've already tried the obvious Filling -> {1 -> {2}} but Mathematica doesn't seem to like that option. Is there any easy way around this or ... Answer There is no built-in filling in SphericalPlot3D . One option is to use ParametricPlot3D to draw the surfaces between the two shells: Manipulate[ Show[SphericalPlot3D[{1, 2 - n}, {θ, 0, Pi}, {ϕ, 0, 1.5 Pi}, PlotPoints -> 15, PlotRange -> {-2.2, 2.2}], ParametricPlot3D[{ r {Sin[t] Cos[1.5 Pi], Sin[t] Sin[1.5 Pi], Cos[t]}, r {Sin[t] Cos[0 Pi], Sin[t] Sin[0 Pi], Cos[t]}}, {r, 1, 2 - n}, {t, 0, Pi}, PlotStyle -> Yellow, Mesh -> {2, 15}]], {n, 0, 1}]

plotting - Plot 4D data with color as 4th dimension

I have a list of 4D data (x position, y position, amplitude, wavelength). I want to plot x, y, and amplitude on a 3D plot and have the color of the points correspond to the wavelength. I have seen many examples using functions to define color but my wavelength cannot be expressed by an analytic function. Is there a simple way to do this? Answer Here a another possible way to visualize 4D data: data = Flatten[Table[{x, y, x^2 + y^2, Sin[x - y]}, {x, -Pi, Pi,Pi/10}, {y,-Pi,Pi, Pi/10}], 1]; You can use the function Point along with VertexColors . Now the points are places using the first three elements and the color is determined by the fourth. In this case I used Hue, but you can use whatever you prefer. Graphics3D[ Point[data[[All, 1 ;; 3]], VertexColors -> Hue /@ data[[All, 4]]], Axes -> True, BoxRatios -> {1, 1, 1/GoldenRatio}]

plotting - Adding a thick curve to a regionplot

Suppose we have the following simple RegionPlot: f[x_] := 1 - x^2 g[x_] := 1 - 0.5 x^2 RegionPlot[{y < f[x], f[x] < y < g[x], y > g[x]}, {x, 0, 2}, {y, 0, 2}] Now I'm trying to change the curve defined by $y=g[x]$ into a thick black curve, while leaving all other boundaries in the plot unchanged. I've tried adding the region $y=g[x]$ and playing with the plotstyle, which didn't work, and I've tried BoundaryStyle, which changed all the boundaries in the plot. Now I'm kinda out of ideas... Any help would be appreciated! Answer With f[x_] := 1 - x^2 g[x_] := 1 - 0.5 x^2 You can use Epilog to add the thick line: RegionPlot[{y < f[x], f[x] < y < g[x], y > g[x]}, {x, 0, 2}, {y, 0, 2}, PlotPoints -> 50, Epilog -> (Plot[g[x], {x, 0, 2}, PlotStyle -> {Black, Thick}][[1]]), PlotStyle -> {Directive[Yellow, Opacity[0.4]], Directive[Pink, Opacity[0.4]],