2023-07-14 16:53:30 +03:00
|
|
|
#include <cstring>
|
|
|
|
#include <iostream>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
#include "config.hh"
|
|
|
|
#include "eval.hh"
|
|
|
|
#include "globals.hh"
|
|
|
|
#include "util.hh"
|
|
|
|
|
|
|
|
#include "nix_api_expr.h"
|
|
|
|
#include "nix_api_expr_internal.h"
|
|
|
|
#include "nix_api_store.h"
|
|
|
|
#include "nix_api_store_internal.h"
|
|
|
|
#include "nix_api_util.h"
|
|
|
|
#include "nix_api_util_internal.h"
|
|
|
|
|
|
|
|
#ifdef HAVE_BOEHMGC
|
2023-07-28 11:49:21 +03:00
|
|
|
#include <mutex>
|
2023-07-14 16:53:30 +03:00
|
|
|
#define GC_INCLUDE_NEW 1
|
|
|
|
#include "gc_cpp.h"
|
|
|
|
#endif
|
|
|
|
|
2023-08-28 17:45:02 +03:00
|
|
|
nix_err nix_libexpr_init(nix_c_context * context)
|
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
{
|
|
|
|
auto ret = nix_libutil_init(context);
|
|
|
|
if (ret != NIX_OK)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
{
|
|
|
|
auto ret = nix_libstore_init(context);
|
|
|
|
if (ret != NIX_OK)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
nix::initGC();
|
|
|
|
}
|
|
|
|
NIXC_CATCH_ERRS
|
2023-07-14 16:53:30 +03:00
|
|
|
}
|
|
|
|
|
2024-01-10 12:58:35 +02:00
|
|
|
nix_err nix_expr_eval_from_string(
|
|
|
|
nix_c_context * context, EvalState * state, const char * expr, const char * path, Value * value)
|
2023-08-28 17:45:02 +03:00
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
try {
|
|
|
|
nix::Expr * parsedExpr = state->state.parseExprFromString(expr, state->state.rootPath(nix::CanonPath(path)));
|
|
|
|
state->state.eval(parsedExpr, *(nix::Value *) value);
|
|
|
|
state->state.forceValue(*(nix::Value *) value, nix::noPos);
|
|
|
|
}
|
|
|
|
NIXC_CATCH_ERRS
|
2023-07-14 16:53:30 +03:00
|
|
|
}
|
|
|
|
|
2024-01-10 12:58:35 +02:00
|
|
|
nix_err nix_value_call(nix_c_context * context, EvalState * state, Value * fn, Value * arg, Value * value)
|
2023-08-28 17:45:02 +03:00
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
try {
|
|
|
|
state->state.callFunction(*(nix::Value *) fn, *(nix::Value *) arg, *(nix::Value *) value, nix::noPos);
|
|
|
|
state->state.forceValue(*(nix::Value *) value, nix::noPos);
|
|
|
|
}
|
|
|
|
NIXC_CATCH_ERRS
|
2023-07-14 16:53:30 +03:00
|
|
|
}
|
|
|
|
|
2024-05-23 20:55:32 +03:00
|
|
|
nix_err nix_value_call_multi(nix_c_context * context, EvalState * state, Value * fn, size_t nargs, Value ** args, Value * value)
|
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
try {
|
|
|
|
state->state.callFunction(*(nix::Value *) fn, nargs, (nix::Value * *)args, *(nix::Value *) value, nix::noPos);
|
|
|
|
state->state.forceValue(*(nix::Value *) value, nix::noPos);
|
|
|
|
}
|
|
|
|
NIXC_CATCH_ERRS
|
|
|
|
}
|
|
|
|
|
2024-01-10 12:58:35 +02:00
|
|
|
nix_err nix_value_force(nix_c_context * context, EvalState * state, Value * value)
|
2023-08-28 17:45:02 +03:00
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
try {
|
|
|
|
state->state.forceValue(*(nix::Value *) value, nix::noPos);
|
|
|
|
}
|
|
|
|
NIXC_CATCH_ERRS
|
2023-07-14 16:53:30 +03:00
|
|
|
}
|
|
|
|
|
2024-01-10 12:58:35 +02:00
|
|
|
nix_err nix_value_force_deep(nix_c_context * context, EvalState * state, Value * value)
|
2023-08-28 17:45:02 +03:00
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
try {
|
|
|
|
state->state.forceValueDeep(*(nix::Value *) value);
|
|
|
|
}
|
|
|
|
NIXC_CATCH_ERRS
|
2023-07-14 16:53:30 +03:00
|
|
|
}
|
|
|
|
|
2024-04-13 18:35:15 +03:00
|
|
|
EvalState * nix_state_create(nix_c_context * context, const char ** lookupPath_c, Store * store)
|
2023-08-28 17:45:02 +03:00
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
try {
|
2024-04-13 18:35:15 +03:00
|
|
|
nix::Strings lookupPath;
|
|
|
|
if (lookupPath_c != nullptr)
|
|
|
|
for (size_t i = 0; lookupPath_c[i] != nullptr; i++)
|
|
|
|
lookupPath.push_back(lookupPath_c[i]);
|
2023-08-28 17:45:02 +03:00
|
|
|
|
2024-04-13 18:35:15 +03:00
|
|
|
return new EvalState{nix::EvalState(nix::LookupPath::parse(lookupPath), store->ptr)};
|
2023-08-28 17:45:02 +03:00
|
|
|
}
|
|
|
|
NIXC_CATCH_ERRS_NULL
|
2023-07-14 16:53:30 +03:00
|
|
|
}
|
|
|
|
|
2024-01-10 12:58:35 +02:00
|
|
|
void nix_state_free(EvalState * state)
|
2023-08-28 17:45:02 +03:00
|
|
|
{
|
|
|
|
delete state;
|
|
|
|
}
|
2023-07-14 16:53:30 +03:00
|
|
|
|
2023-07-28 11:49:21 +03:00
|
|
|
#ifdef HAVE_BOEHMGC
|
|
|
|
std::unordered_map<
|
2023-08-28 17:45:02 +03:00
|
|
|
const void *,
|
|
|
|
unsigned int,
|
|
|
|
std::hash<const void *>,
|
2023-07-28 11:49:21 +03:00
|
|
|
std::equal_to<const void *>,
|
2023-08-28 17:45:02 +03:00
|
|
|
traceable_allocator<std::pair<const void * const, unsigned int>>>
|
2023-07-28 11:49:21 +03:00
|
|
|
nix_refcounts;
|
|
|
|
|
|
|
|
std::mutex nix_refcount_lock;
|
|
|
|
|
2023-08-28 17:45:02 +03:00
|
|
|
nix_err nix_gc_incref(nix_c_context * context, const void * p)
|
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
try {
|
|
|
|
std::scoped_lock lock(nix_refcount_lock);
|
|
|
|
auto f = nix_refcounts.find(p);
|
|
|
|
if (f != nix_refcounts.end()) {
|
|
|
|
f->second++;
|
|
|
|
} else {
|
|
|
|
nix_refcounts[p] = 1;
|
|
|
|
}
|
2023-07-28 17:21:29 +03:00
|
|
|
}
|
2023-08-28 17:45:02 +03:00
|
|
|
NIXC_CATCH_ERRS
|
2023-07-14 16:53:30 +03:00
|
|
|
}
|
|
|
|
|
2023-08-28 17:45:02 +03:00
|
|
|
nix_err nix_gc_decref(nix_c_context * context, const void * p)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
try {
|
|
|
|
std::scoped_lock lock(nix_refcount_lock);
|
|
|
|
auto f = nix_refcounts.find(p);
|
|
|
|
if (f != nix_refcounts.end()) {
|
|
|
|
if (--f->second == 0)
|
|
|
|
nix_refcounts.erase(f);
|
|
|
|
} else
|
|
|
|
throw std::runtime_error("nix_gc_decref: object was not referenced");
|
|
|
|
}
|
|
|
|
NIXC_CATCH_ERRS
|
2023-07-28 11:49:21 +03:00
|
|
|
}
|
|
|
|
|
2023-08-28 17:45:02 +03:00
|
|
|
void nix_gc_now()
|
|
|
|
{
|
|
|
|
GC_gcollect();
|
|
|
|
}
|
2023-07-28 14:47:54 +03:00
|
|
|
|
2023-07-14 16:53:30 +03:00
|
|
|
#else
|
2024-04-04 22:24:42 +03:00
|
|
|
nix_err nix_gc_incref(nix_c_context * context, const void *)
|
2023-08-28 17:45:02 +03:00
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
return NIX_OK;
|
2023-07-28 17:21:29 +03:00
|
|
|
}
|
2024-04-04 22:24:42 +03:00
|
|
|
nix_err nix_gc_decref(nix_c_context * context, const void *)
|
2023-08-28 17:45:02 +03:00
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
context->last_err_code = NIX_OK;
|
|
|
|
return NIX_OK;
|
2023-07-28 17:21:29 +03:00
|
|
|
}
|
2023-07-28 14:47:54 +03:00
|
|
|
void nix_gc_now() {}
|
2023-07-14 16:53:30 +03:00
|
|
|
#endif
|
|
|
|
|
2023-08-28 17:45:02 +03:00
|
|
|
void nix_gc_register_finalizer(void * obj, void * cd, void (*finalizer)(void * obj, void * cd))
|
|
|
|
{
|
2023-07-14 16:53:30 +03:00
|
|
|
#ifdef HAVE_BOEHMGC
|
2023-08-28 17:45:02 +03:00
|
|
|
GC_REGISTER_FINALIZER(obj, finalizer, cd, 0, 0);
|
2023-07-14 16:53:30 +03:00
|
|
|
#endif
|
|
|
|
}
|