All Downloads are FREE. Search and download functionalities are using the official Maven repository.

z3-z3-4.13.0.src.api.dotnet.NativeModel.cs Maven / Gradle / Ivy

The newest version!
/*++
Copyright (c) 2012 Microsoft Corporation

Module Name:

    NativeModel.cs

Abstract:

    Z3 Managed API: Models
    Native interface to model objects.    

Author:

    Christoph Wintersteiger (cwinter) 2012-03-21
    Nikolaj Bjorner (nbjorner) 2022-03-01

Notes:
    
--*/

using System;
using System.Collections.Generic;
using System.Linq;
using System.Diagnostics;

namespace Microsoft.Z3
{
    using Z3_ast = System.IntPtr;
    using Z3_func_decl = System.IntPtr;
    using Z3_sort = System.IntPtr;

    /// 
    /// A Model contains interpretations (assignments) of constants and functions. 
    /// 
    public class NativeModel : IDisposable
    {

        /// 
        /// Retrieves the interpretation (the assignment) of  in the model. 
        /// 
        /// A Constant
        /// An expression if the constant has an interpretation in the model, null otherwise.
        public Z3_ast ConstInterp(Z3_ast a) => ConstFuncInterp(Native.Z3_get_app_decl(ntvContext.nCtx, a));

        /// 
        /// Retrieves the interpretation (the assignment) of  in the model. 
        /// 
        /// A function declaration of zero arity
        /// An expression if the function has an interpretation in the model, null otherwise.    
        public Z3_ast ConstFuncInterp(Z3_func_decl f)
        {
            if (Native.Z3_get_arity(ntvContext.nCtx, f) != 0)
                throw new Z3Exception("Non-zero arity functions have FunctionInterpretations as a model. Use FuncInterp.");

            return Native.Z3_model_get_const_interp(ntvContext.nCtx, NativeObject, f);
        }

        /// 
        /// Retrieves the interpretation (the assignment) of a non-constant  in the model. 
        /// 
        /// A function declaration of non-zero arity
        /// A FunctionInterpretation if the function has an interpretation in the model, null otherwise. 
        public NativeFuncInterp FuncInterp(Z3_func_decl f)
        {
            Z3_sort_kind sk = (Z3_sort_kind)Native.Z3_get_sort_kind(ntvContext.nCtx, Native.Z3_get_range(ntvContext.nCtx, f));

            if (Native.Z3_get_arity(ntvContext.nCtx, f) == 0)
            {
                IntPtr n = Native.Z3_model_get_const_interp(ntvContext.nCtx, NativeObject, f);

                if (sk == Z3_sort_kind.Z3_ARRAY_SORT)
                {
                    if (n == IntPtr.Zero)
                        return null;
                    else
                    {
                        if (Native.Z3_is_as_array(ntvContext.nCtx, n) == 0)
                            throw new Z3Exception("Argument was not an array constant");
                        var fd = Native.Z3_get_as_array_func_decl(ntvContext.nCtx, n);
                        return new NativeFuncInterp(ntvContext, this, f, fd);
                    }
                }
                else
                {
                    throw new Z3Exception("Constant functions do not have a function interpretation; use ConstInterp");
                }
            }
            else
            {
                IntPtr n = Native.Z3_model_get_func_interp(ntvContext.nCtx, NativeObject, f);
                if (n == IntPtr.Zero)
                    return null;
                else
                    return new NativeFuncInterp(ntvContext, this, f, n);
            }
        }



        /// 
        /// The number of constants that have an interpretation in the model.
        /// 
        public uint NumConsts
        {
            get { return Native.Z3_model_get_num_consts(ntvContext.nCtx, NativeObject); }
        }


        /// 
        /// The function declarations of the constants in the model.
        /// 
        public Z3_func_decl[] ConstDecls
        {
            get
            {

                uint n = NumConsts;
                Z3_func_decl[] res = new Z3_func_decl[n];
                for (uint i = 0; i < n; i++)
                    res[i] = Native.Z3_model_get_const_decl(ntvContext.nCtx, NativeObject, i);
                return res;
            }
        }


        /// 
        /// Enumerate constants in model.
        /// 
        public IEnumerable> Consts
        {
            get
            {
                uint nc = NumConsts;
                for (uint i = 0; i < nc; ++i)
                {
                    var f = Native.Z3_model_get_const_decl(ntvContext.nCtx, NativeObject, i);
                    IntPtr n = Native.Z3_model_get_const_interp(ntvContext.nCtx, NativeObject, f);
                    if (n == IntPtr.Zero) continue;
                    yield return new KeyValuePair(f, n);
                }
            }
        }

        /// 
        /// The number of function interpretations in the model.
        /// 
        public uint NumFuncs
        {
            get { return Native.Z3_model_get_num_funcs(ntvContext.nCtx, NativeObject); }
        }

        /// 
        /// The function declarations of the function interpretations in the model.
        /// 
        public Z3_func_decl[] FuncDecls
        {
            get
            {

                uint n = NumFuncs;
                Z3_func_decl[] res = new Z3_func_decl[n];
                for (uint i = 0; i < n; i++)
                    res[i] = Native.Z3_model_get_func_decl(ntvContext.nCtx, NativeObject, i);
                return res;
            }
        }

        /// 
        /// All symbols that have an interpretation in the model.
        /// 
        public Z3_func_decl[] Decls
        {
            get
            {

                uint nFuncs = NumFuncs;
                uint nConsts = NumConsts;
                uint n = nFuncs + nConsts;
                Z3_func_decl[] res = new Z3_func_decl[n];
                for (uint i = 0; i < nConsts; i++)
                    res[i] = Native.Z3_model_get_const_decl(ntvContext.nCtx, NativeObject, i);
                for (uint i = 0; i < nFuncs; i++)
                    res[nConsts + i] = Native.Z3_model_get_func_decl(ntvContext.nCtx, NativeObject, i);
                return res;
            }
        }

        /// 
        /// A ModelEvaluationFailedException is thrown when an expression cannot be evaluated by the model.
        /// 
        public class ModelEvaluationFailedException : Z3Exception
        {
            /// 
            /// An exception that is thrown when model evaluation fails.
            /// 
            public ModelEvaluationFailedException() : base() { }
        }

        /// 
        /// Evaluates the expression  in the current model.
        /// 
        /// 
        /// This function may fail if  contains quantifiers, 
        /// is partial (MODEL_PARTIAL enabled), or if  is not well-sorted.
        /// In this case a ModelEvaluationFailedException is thrown.
        /// 
        /// An expression
        /// 
        /// When this flag is enabled, a model value will be assigned to any constant 
        /// or function that does not have an interpretation in the model.
        /// 
        /// The evaluation of  in the model.        
        public Z3_ast Eval(Z3_ast t, bool completion = false)
        {

            IntPtr v = IntPtr.Zero;
            if (Native.Z3_model_eval(ntvContext.nCtx, NativeObject, t, (byte)(completion ? 1 : 0), ref v) == (byte)0)
                throw new ModelEvaluationFailedException();
            else
                return v;
        }

        /// 
        /// Alias for Eval.
        ///         
        public Z3_ast Evaluate(Z3_ast t, bool completion = false) => Eval(t, completion);

        /// 
        /// Evaluate expression to a double, assuming it is a numeral already.
        /// 
        public double Double(Z3_ast t)
        {
            var r = Eval(t, true);
            return Native.Z3_get_numeral_double(ntvContext.nCtx, r);
        }

        /// 
        /// An array value obtained by untangling a model assignment.
        /// 
        public class ArrayValue
        {
            /// 
            /// One dimensional array of indices where the array is updated
            /// 
            public KeyValuePair[] Updates;

            /// 
            /// default Else case
            /// 
            public Z3_ast Else;

            /// 
            /// Domain for array
            /// Updates.Keys
            /// 
            public Z3_ast[] Domain;

            /// 
            /// Range for array
            /// Updates.Values
            /// 
            public Z3_ast[] Range;
        }

        /// 
        /// Convert the interpretation of t into a sequence of array updates
        /// 
        /// 
        /// 
        /// null if the argument does evaluate to a sequence of stores to an array
        public bool TryGetArrayValue(Z3_ast t, out ArrayValue result)
        {
            var r = Eval(t, true);
            // check that r is a sequence of store over a constant default array.
            var updates = new Dictionary();
            result = null;
            while (true)
            {
                if (ntvContext.GetAstKind(r) != Z3_ast_kind.Z3_APP_AST)
                    return false;
                Z3_func_decl f = ntvContext.GetAppDecl(r);
                var kind = ntvContext.GetDeclKind(f);
                if (kind == Z3_decl_kind.Z3_OP_CONST_ARRAY)
                {
                    result = new ArrayValue();
                    result.Else = ntvContext.GetAppArg(r, 0);
                    result.Updates = updates.ToArray();
                    result.Domain = updates.Keys.ToArray();
                    result.Range = updates.Values.ToArray();
                    return true;
                }
                else if (kind == Z3_decl_kind.Z3_OP_STORE)
                {
                    Debug.Assert(ntvContext.GetNumArgs(r) == 3);
                    updates[ntvContext.GetAppArg(r, 1)] = ntvContext.GetAppArg(r, 2);
                    r = ntvContext.GetAppArg(r, 0);
                }
                else
                {
                    return false;
                }
            }
        }

        /// 
        /// The number of uninterpreted sorts that the model has an interpretation for.
        ///     
        public uint NumSorts { get { return Native.Z3_model_get_num_sorts(ntvContext.nCtx, NativeObject); } }


        /// 
        /// The uninterpreted sorts that the model has an interpretation for. 
        /// 
        /// 
        /// Z3 also provides an interpretation for uninterpreted sorts used in a formula.
        /// The interpretation for a sort is a finite set of distinct values. We say this finite set is
        /// the "universe" of the sort.
        /// 
        /// 
        public Z3_sort[] Sorts
        {
            get
            {

                uint n = NumSorts;
                Z3_sort[] res = new Z3_sort[n];
                for (uint i = 0; i < n; i++)
                    res[i] = Native.Z3_model_get_sort(ntvContext.nCtx, NativeObject, i);
                return res;
            }
        }


        /// 
        /// Conversion of models to strings. 
        /// 
        /// A string representation of the model.
        public override string ToString()
        {
            return Native.Z3_model_to_string(ntvContext.nCtx, NativeObject);
        }


        IntPtr NativeObject;
        NativeContext ntvContext;

        internal NativeModel(NativeContext ctx, IntPtr obj)
        {
            ntvContext = ctx;
            NativeObject = obj;
            Debug.Assert(ctx != null);
            Native.Z3_model_inc_ref(ctx.nCtx, obj);
        }


        /// 
        /// Finalizer.
        /// 
        ~NativeModel()
        {
            Dispose();
        }

        /// 
        /// Disposes of the underlying native Z3 object.
        /// 
        public void Dispose()
        {
            if (NativeObject != IntPtr.Zero)
            {
                Native.Z3_model_dec_ref(ntvContext.nCtx, NativeObject);
                NativeObject = IntPtr.Zero;
            }
            GC.SuppressFinalize(this);
        }


    }
}







© 2015 - 2024 Weber Informatics LLC | Privacy Policy