ComputationalResources

Build Status

codecov.io

A Julia package for choosing resources (hardware, libraries, algorithms) for performing computations. It exports a set of types that allow you to dispatch to different methods depending on the selected resources. It also includes simple facilities that help package authors manage the loading of code in a way that depends on resource availability and user choice.

Resources

This package exports the following types of resources:

  • CPU1 (single-threaded computation)
  • CPUThreads (multi-threaded computation)
  • CPUProcesses (multi-process computation)
  • ArrayFireLibs (using the ArrayFire package)
  • CUDALibs (GPU computation using NVIDIA's CUDA libraries)
  • OpenCLLibs (GPU computation using the OpenCL libraries)

Algorithm selection is performed by passing a resource instance as an argument (conventionally, the first argument) to a function, where CPU1() is the typical default. Instances of these types may optionally store additional settings that you can customize; for example you could define a TimeOut type:

struct TimeOut
    seconds::Float64
end

and then call some algorithm as

optimize(CPU1(TimeOut(3)), f, x)

As a package author, you could write optimize to check for the timeout value and terminate early once this time has been exceeded.

Usage as a user

Begin your session with

using ComputationalResources

Then choose any resources you have available, for example:

addresource(ArrayFireLibs)

It's important to do this before you load any packages supporting ArrayFire-specific implementations; otherwise, calls such as

filter(ArrayFireLibs(), b, a, data)

are likely to throw a MethodError because the relevant specializations will not have been loaded.

If you'd like to make your selection of available resources automatic, you can add such lines to your .juliarc.jl file.

Usage as a package author

You can make the loading of code dependent upon the resources selected by the user. We'll use the "Dummy" package as an example (see also the test/packages folder for additional examples). This package could have the following file structure:

src/
  Dummy.jl
  DummyAF.jl
  ...
test/
  ...

where ... means additional files. Dummy.jl might start like this:

module Dummy

using ComputationalResources

# You need an __init__ function that will manage the loading of
# sub-modules that implement specializations for different
# computational resources
function __init__()
    # Enable `using` to load additional modules in this folder
    push!(LOAD_PATH, dirname(@__FILE__))
    # Now check for any resources that your package supports
    if haveresource(ArrayFireLibs)
        # User has indicated support for the ArrayFire libraries, so load your relevant code
        @eval using DummyAF
    end
    # Put additional resource checks here
    # Don't forget to clean up!
    pop!(LOAD_PATH)
end

# Now define the methods you'll export, using single-threaded CPU
# computations as the "foundation"
function foo(resource::CPU1, args...)
    # awesome algorithm goes here
end

# Typically you should select a default resource
foo(args...) = foo(CPU1(), args...)

end

Your DummyAF module is implemented in DummyAF.jl, which might look like this:

module DummyAF

using ComputationalResources, Dummy, ArrayFire

function Dummy.foo(resource::ArrayFireLibs, args...)
    # specialization for the same computation, but using the ArrayFire libraries instead
end

end

Note that the ArrayFire package was loaded by DummyAF but not by Dummy; as a consequence, users who do not have this package installed will not experience any errors. Assuming it's optional, you should not add ArrayFire to your package's REQUIRE file.