Skip to content

Commit

Permalink
Use metaprogramming for generic functions in core_interface.jl
Browse files Browse the repository at this point in the history
  • Loading branch information
amontoison committed Aug 19, 2024
1 parent 781e33f commit b6ddce9
Show file tree
Hide file tree
Showing 12 changed files with 2,653 additions and 2,726 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ jobs:
with:
compiler: 'gcc'
version: '11'
if: matrix.os == 'macos-latest' || matrix.os == 'macos-13'
if: matrix.os != 'ubuntu-latest'
- uses: julia-actions/setup-julia@v2
with:
version: ${{ matrix.version }}
Expand Down
2 changes: 1 addition & 1 deletion docs/src/core.md
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ For each of those, we dropped the `cutest_`, so the functions `cutest_ufn` and
To use then you have to convert the types using `Cint` and `Cdouble`, and
pass arrays because of the underlying pointers in Fortran.
In practice, there isn't much improvement in calling these or `ccall`s, except
for the use of the internal `cutest_lib`.
for the use of the internal `cutest_lib_double`.

**Only use these functions if you really know what you're doing.**

Expand Down
44 changes: 43 additions & 1 deletion gen/wrapper.jl
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ function main()

args = get_default_args()
push!(args, "-I$include_dir")
push!(args, "-DREAL_128")

ctx = create_context(headers, args, options)
build!(ctx)
Expand All @@ -32,19 +33,60 @@ function main()
code = replace(code, "Ptr{ip_}" => "Ptr{Cint}")
code = replace(code, "Ptr{ipc_}" => "Ptr{Cint}")

# errors in cutest.h
code = replace(code, "cutest_cint_csgrp_" => "cutest_csgrp_")
code = replace(code, "cutest_cint_csjp_" => "cutest_csjp_")
code = replace(code, "cutest_cint_csgrshp_" => "cutest_csgrshp_")
code = replace(code, "cutest_cint_cchprodsp_" => "cutest_cchprodsp_")

blocks = split(code, "end\n")
nblocks = length(blocks)
code = ""
for (index, block) in enumerate(blocks)
if contains(block, "function")
fname = split(split(block, "function ")[2], "(")[1]
ptr = "ptr_$(fname) = Libdl.dlsym(cutest_lib, :$(fname))"
if contains(block, "_s(") || contains(block, "_s_(")
ptr = "ptr_$(fname) = Libdl.dlsym(cutest_lib_single, :$(fname))"
elseif contains(block, "_q(") || contains(block, "_q_(")
ptr = "ptr_$(fname) = Libdl.dlsym(cutest_lib_quadruple, :$(fname))"
else
ptr = "ptr_$(fname) = Libdl.dlsym(cutest_lib_double, :$(fname))"
end
block = replace(block, " @ccall libcutest.$fname" => " $ptr\n @ccall \$ptr_$(fname)")
end
code = code * block
(index < nblocks) && (code = code * "end\n")
end

# Add wrappers of "fortran_open" and "fortran_close" for single and quadruple precisison
blocks = split(code, "end\n")
nblocks = length(blocks)
code = ""
for (index, block) in enumerate(blocks)
code = code * block
(index < nblocks) && (code = code * "end\n")
for routine in ("fortran_open_", "fortran_close_")
# add the routines `fortran_open_s_` and `fortran_close_s_`
if contains(block, routine)
block_single = block
block_single = replace(block_single, "double" => "single")
block_single = replace(block_single, routine => "$(routine)s_")
block_single = replace(block_single, ":$(routine)s_" => ":$(routine)")
code = code * block_single
(index < nblocks) && (code = code * "end\n")
end
# Add the routines `fortran_open_q_` and `fortran_close_q_`
if contains(block, routine)
block_quadruple = block
block_quadruple = replace(block_quadruple, "double" => "quadruple")
block_quadruple = replace(block_quadruple, routine => "$(routine)q_")
block_quadruple = replace(block_quadruple, ":$(routine)q_" => ":$(routine)")
code = code * block_quadruple
(index < nblocks) && (code = code * "end\n")
end
end
end

write(path, code)
format_file(path, YASStyle())

Expand Down
26 changes: 14 additions & 12 deletions src/CUTEst.jl
Original file line number Diff line number Diff line change
@@ -1,8 +1,4 @@
#See JuliaSmoothOptimizers/NLPModels.jl/issues/113
__precompile__()

# Using CUTEst from Julia.

module CUTEst

using CUTEst_jll
Expand All @@ -14,8 +10,12 @@ using NLPModels
import Libdl.dlsym

# Only one problem can be interfaced at any given time.
global cutest_instances = 0
global cutest_lib = C_NULL
global cutest_instances_single = 0
global cutest_instances_double = 0
global cutest_instances_quadruple = 0
global cutest_lib_single = C_NULL
global cutest_lib_double = C_NULL
global cutest_lib_quadruple = C_NULL

export CUTEstModel, sifdecoder, build_libsif, set_mastsif

Expand All @@ -28,12 +28,14 @@ const cutest_problems_path = joinpath(dirname(@__FILE__), "..", "deps", "files")
isdir(cutest_problems_path) || mkpath(cutest_problems_path)

function __init__()
if success(`bash -c "type gfortran"`)
@static Sys.isapple() ? (global libgfortran = []) :
(global libgfortran = [strip(read(`gfortran --print-file libgfortran.so`, String))])
else
@error "gfortran is not installed. Please install it and try again."
return
if !Sys.iswindows()
if success(`bash -c "type gfortran"`)
@static Sys.isapple() ? (global libgfortran = []) :
(global libgfortran = [strip(read(`gfortran --print-file libgfortran.so`, String))])
else
@error "gfortran is not installed. Please install it and try again."
return
end
end

# set default MASTSIF location if the user hasn't set it already
Expand Down
Loading

0 comments on commit b6ddce9

Please sign in to comment.