Skip to main content

Code generation with CVXPY

Project description

CVXPYgen: Code generation with CVXPY

CVXPYgen takes a convex optimization problem family modeled with CVXPY and generates a custom solver implementation in C. This generated solver is specific to the problem family and accepts different parameter values. In particular, this solver is suitable for deployment on embedded systems. In addition, CVXPYgen creates a Python wrapper for prototyping and desktop (non-embedded) applications.

An overview of CVXPYgen can be found in our slides and manuscript.

CVXPYgen accepts CVXPY problems that are compliant with Disciplined Convex Programming (DCP). DCP is a system for constructing mathematical expressions with known curvature from a given library of base functions. CVXPY uses DCP to ensure that the specified optimization problems are convex. In addition, problems need to be modeled according to Disciplined Parametrized Programming (DPP). Solving a DPP-compliant problem repeatedly for different values of the parameters can be much faster than repeatedly solving a new problem.

For now, CVXPYgen is a separate module, until it will be integrated into CVXPY. As of today, CVXPYgen works with linear, quadratic, and second-order cone programs. It also supports differentiating through quadratic programs and computing an explicit solution to linear and quadratic programs.

This package has similar functionality as the package cvxpy_codegen, which appears to be unsupported.

Installation

pip install cvxpygen

If you wish to use the Clarabel solver, you need to install Rust and Eigen.

The example notebooks located in examples/ require matplotlib.

Windows users: CVXPYgen is tested with Visual Studio 2019 and 2022, newer and older versions might work as well.

Example

We define a simple 'nonnegative least squares' problem, generate code for it, and solve the problem with example parameter values.

1. Generate Code

Let's step through the first part of examples/main.py. Define a convex optimization problem the way you are used to with CVXPY. Everything that is described as cp.Parameter() is assumed to be changing between multiple solves. For constant properties, use cp.Constant().

import cvxpy as cp

m, n = 3, 2
x = cp.Variable(n, name='x')
A = cp.Parameter((m, n), name='A', sparsity=((0, 0, 1), (0, 1, 1)))
b = cp.Parameter(m, name='b')
problem = cp.Problem(cp.Minimize(cp.sum_squares(A @ x - b)), [x >= 0])

Specify the name attribute for variables and parameters to recognize them after generating code. The attribute sparsity is a tuple of row and column indices of the nonzero entries of matrix A. Parameter sparsity is only taken into account for matrices.

Assign parameter values and test-solve.

import numpy as np

np.random.seed(0)
A.value = np.zeros((m, n))
A.value[0, 0] = np.random.randn()
A.value[0, 1] = np.random.randn()
A.value[1, 1] = np.random.randn()
b.value = np.random.randn(m)
problem.solve()

Generating C code for this problem is as simple as,

from cvxpygen import cpg

cpg.generate_code(problem, code_dir='nonneg_LS', solver='SCS')

where the generated code is stored inside nonneg_LS and the SCS solver is used. Next to the positional argument problem, all keyword arguments for the generate_code() method are summarized below.

Argument Meaning Type Default
code_dir directory for code to be stored in String 'CPG_code'
solver canonical solver to generate code with String CVXPY default
solver_opts options passed to canonical solver Dict None
enable_settings enabled settings that are otherwise locked by embedded solver List of Strings []
unroll unroll loops in canonicalization code Bool False
prefix prefix for unique code symbols when dealing with multiple problems String ''
wrapper compile Python wrapper for CVXPY interface Bool True
gradient enable differentiation (works for linear and quadratic programs) Bool False

You can find an overview of the code generation result in nonneg_LS/README.html.

2. Solve & Compare

As summarized in the second part of examples/main.py, after assigning parameter values, you can solve the problem both conventionally and via the generated code, which is wrapped inside the custom CVXPY solve method cpg_solve.

import time
import sys

# import extension module and register custom CVXPY solve method
from nonneg_LS.cpg_solver import cpg_solve
problem.register_solve('CPG', cpg_solve)

# solve problem conventionally
t0 = time.time()
val = problem.solve(solver='SCS')
t1 = time.time()
print('\nCVXPY\nSolve time: %.3f ms\n' % (1000*(t1-t0)))
print('Primal solution: x = [%.6f, %.6f]\n' % tuple(x.value))
print('Dual solution: d0 = [%.6f, %.6f]\n' % tuple(problem.constraints[0].dual_value))
print('Objective function value: %.6f\n' % val)

# solve problem with C code via python wrapper
t0 = time.time()
val = problem.solve(method='CPG', updated_params=['A', 'b'], verbose=False)
t1 = time.time()
print('\nCVXPYgen\nSolve time: %.3f ms\n' % (1000 * (t1 - t0)))
print('Primal solution: x = [%.6f, %.6f]\n' % tuple(x.value))
print('Dual solution: d0 = [%.6f, %.6f]\n' % tuple(problem.constraints[0].dual_value))
print('Objective function value: %.6f\n' % val)

The argument updated_params specifies which user-defined parameter values are new. If the argument is omitted, all parameter values are assumed to be new. If only a subset of the user-defined parameters have new values, use this argument to speed up the solver.

Most solver settings can be specified as keyword arguments like without code generation. Here, we use verbose=False to suppress printing. The list of changeable settings differs by solver and is documented in <code_dir>/README.html after code generation.

Comparing the standard and codegen methods for this example, both the solutions and objective values are close. Especially for smaller problems like this, the new solve method 'CPG' is significantly faster than solving without code generation.

3. Executable

In the C code, all of your parameters and variables are stored as vectors via Fortran-style flattening (vertical index moves fastest). For example, the (i, j)-th entry of the original matrix with height h will be the i+j*h-th entry of the flattened matrix in C. For sparse parameters, i.e. matrices, the k-th entry of the C array is the k-th nonzero entry encountered when proceeding through the parameter column by column.

Before compiling the example executable, make sure that CMake 3.5 or newer is installed.

On Unix platforms, run the following commands in your terminal to compile and run the program:

cd nonneg_LS/c/build
cmake ..
cmake --build . --target cpg_example
./cpg_example

On Windows, type:

cd nonneg_LS\c\build
cmake ..
cmake --build . --target cpg_example --config release
Release\cpg_example

Differentiating through problems

CVXPYgen supports differentiating through quadratic programs. To enable this feature, set gradient=True when generating code. You can use the generated code together with CVXPYlayers as

cpg.generate_code(problem, code_dir='code_diff', gradient=True)

from code_diff.cpg_solver import forward, backward
from cvxpylayers.torch import CvxpyLayer

layer = CvxpyLayer(problem, parameters=[A, b], variables=[x], custom_method=(forward, backward))

See our manuscript for more details and examples/paper_grad for three practical examples (from our manuscript), in the areas of machine learning, control, and finance.

Explicitly solving problems

For quadratic programs in which the coefficients of the linear objective terms and the righthand side of the constraints are affine functions of a parameter, the solution is a piecewise affine function of the parameter.

The number of (polyhedral) regions in the solution map can grow exponentially in problem size (specifically, the number of inequality constraints), but when the number of regions is moderate, a so-called explicit solver is practical. Such a solver computes the coefficients of the affine functions and the linear inequalities defining the polyhedral regions offline; to solve a problem instance online it simply evaluates this explicit solution map. Potential advantages of an explicit solver over a more general purpose iterative solver can include transparency, interpretability, reliability, and speed.

CVXPYgen can generate such explicit solvers. To enable this feature, set solver='explicit' when generating code. By default, only the primal solution is computed. To also compute the dual solution, pass solver_opts={'dual': True}. You can choose to store the explicit solution in half precision (instead of single precision), by setting 'fp16': True in solver_opts. Limits on parameters are encouraged and can be represented as standard CVXPY constraints. As of now, we support simple bounds of the form [l <= p, p <= u] where p is a cp.Parameter() and l and u are constants.

See our manuscript for more details. You can control the maximum number of floating point numbers in the explicit solution via 'max_floats' (default is 1e6) and the maximum number of regions via 'max_regions' (default is 500) in the solver_opts dict.

CVXPYgen uses PDAQP to construct explicit solutions.

Tests

To run tests, install pytest via

conda install pytest

and execute:

cd tests
pytest

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

cvxpygen-0.6.5.tar.gz (619.1 kB view details)

Uploaded Source

Built Distribution

cvxpygen-0.6.5-py3-none-any.whl (748.7 kB view details)

Uploaded Python 3

File details

Details for the file cvxpygen-0.6.5.tar.gz.

File metadata

  • Download URL: cvxpygen-0.6.5.tar.gz
  • Upload date:
  • Size: 619.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.11.0

File hashes

Hashes for cvxpygen-0.6.5.tar.gz
Algorithm Hash digest
SHA256 306757011965843a629666f34ebcf23ac414aa1173e366d31de9569f7235045a
MD5 deac05c1f516ad81798d60030914a0f7
BLAKE2b-256 10a8fb5a27d5288c5028a672538acde37cc6395f7ed7a0c946cd20c81156282f

See more details on using hashes here.

File details

Details for the file cvxpygen-0.6.5-py3-none-any.whl.

File metadata

  • Download URL: cvxpygen-0.6.5-py3-none-any.whl
  • Upload date:
  • Size: 748.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.11.0

File hashes

Hashes for cvxpygen-0.6.5-py3-none-any.whl
Algorithm Hash digest
SHA256 c1fd5c2effd7c61257489db5497dc090e7127ab2880917c193d9a7dca6725625
MD5 6d33411caf52fa8d5c0dfe9e36d25f1d
BLAKE2b-256 42ca80f57aefdde5f0f56c45f0fdb0523ce8e5b1a7232625d748f8e4d072cf73

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page