mirror of
https://github.com/Z3Prover/z3
synced 2025-04-15 05:18:44 +00:00
308 lines
9 KiB
C#
308 lines
9 KiB
C#
/*++
|
|
Copyright (c) 2012 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
Optimize.cs
|
|
|
|
Abstract:
|
|
|
|
Z3 Managed API: Optimizes
|
|
|
|
Author:
|
|
|
|
Nikolaj Bjorner (nbjorner) 2013-12-03
|
|
|
|
Notes:
|
|
|
|
--*/
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics.Contracts;
|
|
|
|
namespace Microsoft.Z3
|
|
{
|
|
/// <summary>
|
|
/// Object for managing optimizization context
|
|
/// </summary>
|
|
[ContractVerification(true)]
|
|
public class Optimize : Z3Object
|
|
{
|
|
/// <summary>
|
|
/// A string that describes all available optimize solver parameters.
|
|
/// </summary>
|
|
public string Help
|
|
{
|
|
get
|
|
{
|
|
Contract.Ensures(Contract.Result<string>() != null);
|
|
return Native.Z3_optimize_get_help(Context.nCtx, NativeObject);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the optimize solver parameters.
|
|
/// </summary>
|
|
public Params Parameters
|
|
{
|
|
set
|
|
{
|
|
Contract.Requires(value != null);
|
|
Context.CheckContextMatch(value);
|
|
Native.Z3_optimize_set_params(Context.nCtx, NativeObject, value.NativeObject);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Retrieves parameter descriptions for Optimize solver.
|
|
/// </summary>
|
|
public ParamDescrs ParameterDescriptions
|
|
{
|
|
get { return new ParamDescrs(Context, Native.Z3_optimize_get_param_descrs(Context.nCtx, NativeObject)); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Assert a constraint (or multiple) into the optimize solver.
|
|
/// </summary>
|
|
public void Assert(params BoolExpr[] constraints)
|
|
{
|
|
Contract.Requires(constraints != null);
|
|
Contract.Requires(Contract.ForAll(constraints, c => c != null));
|
|
|
|
Context.CheckContextMatch(constraints);
|
|
foreach (BoolExpr a in constraints)
|
|
{
|
|
Native.Z3_optimize_assert(Context.nCtx, NativeObject, a.NativeObject);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Alias for Assert.
|
|
/// </summary>
|
|
public void Add(params BoolExpr[] constraints)
|
|
{
|
|
Assert(constraints);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Handle to objectives returned by objective functions.
|
|
/// </summary>
|
|
public class Handle
|
|
{
|
|
Optimize opt;
|
|
uint handle;
|
|
internal Handle(Optimize opt, uint h)
|
|
{
|
|
this.opt = opt;
|
|
this.handle = h;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Retrieve a lower bound for the objective handle.
|
|
/// </summary>
|
|
public ArithExpr Lower
|
|
{
|
|
get { return opt.GetLower(handle); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Retrieve an upper bound for the objective handle.
|
|
/// </summary>
|
|
public ArithExpr Upper
|
|
{
|
|
get { return opt.GetUpper(handle); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Retrieve the value of an objective.
|
|
/// </summary>
|
|
public ArithExpr Value
|
|
{
|
|
get { return Lower; }
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Assert soft constraint
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Return an objective which associates with the group of constraints.
|
|
/// </remarks>
|
|
public Handle AssertSoft(BoolExpr constraint, uint weight, string group)
|
|
{
|
|
Context.CheckContextMatch(constraint);
|
|
Symbol s = Context.MkSymbol(group);
|
|
return new Handle(this, Native.Z3_optimize_assert_soft(Context.nCtx, NativeObject, constraint.NativeObject, weight.ToString(), s.NativeObject));
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Check satisfiability of asserted constraints.
|
|
/// Produce a model that (when the objectives are bounded and
|
|
/// don't use strict inequalities) meets the objectives.
|
|
/// </summary>
|
|
///
|
|
public Status Check()
|
|
{
|
|
Z3_lbool r = (Z3_lbool)Native.Z3_optimize_check(Context.nCtx, NativeObject);
|
|
switch (r)
|
|
{
|
|
case Z3_lbool.Z3_L_TRUE:
|
|
return Status.SATISFIABLE;
|
|
case Z3_lbool.Z3_L_FALSE:
|
|
return Status.UNSATISFIABLE;
|
|
default:
|
|
return Status.UNKNOWN;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a backtracking point.
|
|
/// </summary>
|
|
/// <seealso cref="Pop"/>
|
|
public void Push()
|
|
{
|
|
Native.Z3_optimize_push(Context.nCtx, NativeObject);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Backtrack one backtracking point.
|
|
/// </summary>
|
|
/// <remarks>Note that an exception is thrown if Pop is called without a corresponding <c>Push</c></remarks>
|
|
/// <seealso cref="Push"/>
|
|
public void Pop()
|
|
{
|
|
Native.Z3_optimize_pop(Context.nCtx, NativeObject);
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// The model of the last <c>Check</c>.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// The result is <c>null</c> if <c>Check</c> was not invoked before,
|
|
/// if its results was not <c>SATISFIABLE</c>, or if model production is not enabled.
|
|
/// </remarks>
|
|
public Model Model
|
|
{
|
|
get
|
|
{
|
|
IntPtr x = Native.Z3_optimize_get_model(Context.nCtx, NativeObject);
|
|
if (x == IntPtr.Zero)
|
|
return null;
|
|
else
|
|
return new Model(Context, x);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Declare an arithmetical maximization objective.
|
|
/// Return a handle to the objective. The handle is used as
|
|
/// to retrieve the values of objectives after calling Check.
|
|
/// </summary>
|
|
public Handle MkMaximize(ArithExpr e)
|
|
{
|
|
return new Handle(this, Native.Z3_optimize_maximize(Context.nCtx, NativeObject, e.NativeObject));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Declare an arithmetical minimization objective.
|
|
/// Similar to MkMaximize.
|
|
/// </summary>
|
|
public Handle MkMinimize(ArithExpr e)
|
|
{
|
|
return new Handle(this, Native.Z3_optimize_minimize(Context.nCtx, NativeObject, e.NativeObject));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Retrieve a lower bound for the objective handle.
|
|
/// </summary>
|
|
private ArithExpr GetLower(uint index)
|
|
{
|
|
return (ArithExpr)Expr.Create(Context, Native.Z3_optimize_get_lower(Context.nCtx, NativeObject, index));
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Retrieve an upper bound for the objective handle.
|
|
/// </summary>
|
|
private ArithExpr GetUpper(uint index)
|
|
{
|
|
return (ArithExpr)Expr.Create(Context, Native.Z3_optimize_get_upper(Context.nCtx, NativeObject, index));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return a string the describes why the last to check returned unknown
|
|
/// </summary>
|
|
public String getReasonUnknown()
|
|
{
|
|
Contract.Ensures(Contract.Result<string>() != null);
|
|
return Native.Z3_optimize_get_reason_unknown(Context.nCtx, NativeObject);
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Print the context to a string (SMT-LIB parseable benchmark).
|
|
/// </summary>
|
|
public override string ToString()
|
|
{
|
|
return Native.Z3_optimize_to_string(Context.nCtx, NativeObject);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Optimize statistics.
|
|
/// </summary>
|
|
public Statistics Statistics
|
|
{
|
|
get
|
|
{
|
|
Contract.Ensures(Contract.Result<Statistics>() != null);
|
|
|
|
return new Statistics(Context, Native.Z3_optimize_get_statistics(Context.nCtx, NativeObject));
|
|
}
|
|
}
|
|
|
|
|
|
#region Internal
|
|
internal Optimize(Context ctx, IntPtr obj)
|
|
: base(ctx, obj)
|
|
{
|
|
Contract.Requires(ctx != null);
|
|
}
|
|
internal Optimize(Context ctx)
|
|
: base(ctx, Native.Z3_mk_optimize(ctx.nCtx))
|
|
{
|
|
Contract.Requires(ctx != null);
|
|
}
|
|
|
|
internal class DecRefQueue : IDecRefQueue
|
|
{
|
|
public DecRefQueue() : base() { }
|
|
public DecRefQueue(uint move_limit) : base(move_limit) { }
|
|
internal override void IncRef(Context ctx, IntPtr obj)
|
|
{
|
|
Native.Z3_optimize_inc_ref(ctx.nCtx, obj);
|
|
}
|
|
|
|
internal override void DecRef(Context ctx, IntPtr obj)
|
|
{
|
|
Native.Z3_optimize_dec_ref(ctx.nCtx, obj);
|
|
}
|
|
};
|
|
|
|
internal override void IncRef(IntPtr o)
|
|
{
|
|
Context.Optimize_DRQ.IncAndClear(Context, o);
|
|
base.IncRef(o);
|
|
}
|
|
|
|
internal override void DecRef(IntPtr o)
|
|
{
|
|
Context.Optimize_DRQ.Add(o);
|
|
base.DecRef(o);
|
|
}
|
|
#endregion
|
|
}
|
|
}
|