typedef augeas *augeas_t;
+/* Map C aug_errcode_t to OCaml error_code. */
+static const int error_map[] = {
+ /* AugErrInternal */ AUG_EINTERNAL,
+ /* AugErrPathX */ AUG_EPATHX,
+ /* AugErrNoMatch */ AUG_ENOMATCH,
+ /* AugErrMMatch */ AUG_EMMATCH,
+ /* AugErrSyntax */ AUG_ESYNTAX,
+ /* AugErrNoLens */ AUG_ENOLENS,
+ /* AugErrMXfm */ AUG_EMXFM,
+ /* AugErrNoSpan */ AUG_ENOSPAN,
+ /* AugErrMvDesc */ AUG_EMVDESC,
+ /* AugErrCmdRun */ AUG_ECMDRUN,
+ /* AugErrBadArg */ AUG_EBADARG,
+ /* AugErrLabel */ AUG_ELABEL,
+ /* AugErrCpDesc */ AUG_ECPDESC,
+};
+static const int error_map_len = sizeof error_map / sizeof error_map[0];
+
/* Raise an Augeas.Error exception. */
static void
-raise_error (const char *msg)
+raise_error (augeas_t t, const char *msg)
+{
+ value *exn = caml_named_value ("Augeas.Error");
+ value args[4];
+ const int code = aug_error (t);
+ const char *aug_err_minor;
+ const char *aug_err_details;
+ int ocaml_code = -1;
+ int i;
+
+ if (code == AUG_ENOMEM)
+ caml_raise_out_of_memory ();
+
+ aug_err_minor = aug_error_minor_message (t);
+ aug_err_details = aug_error_details (t);
+
+ for (i = 0; i < error_map_len; ++i)
+ if (error_map[i] == code) {
+ ocaml_code = i;
+ break;
+ }
+
+ if (ocaml_code != -1)
+ args[0] = Val_int (ocaml_code);
+ else {
+ args[0] = caml_alloc (1, 0);
+ Store_field (args[0], 0, Val_int (code));
+ }
+ args[1] = caml_copy_string (msg);
+ args[2] = caml_copy_string (aug_err_minor ? : "");
+ args[3] = caml_copy_string (aug_err_details ? : "");
+
+ caml_raise_with_args (*exn, 4, args);
+}
+
+static void
+raise_init_error (const char *msg)
{
- caml_raise_with_string (*caml_named_value ("Augeas.Error"), msg);
+ value *exn = caml_named_value ("Augeas.Error");
+ value args[4];
+
+ args[0] = caml_alloc (1, 0);
+ Store_field (args[0], 0, Val_int (-1));
+ args[1] = caml_copy_string (msg);
+ args[2] = caml_copy_string ("augeas initialization failed");
+ args[3] = caml_copy_string ("");
+
+ caml_raise_with_args (*exn, 4, args);
}
/* Map OCaml flags to C flags. */
t = aug_init (root, loadpath, flags);
if (t == NULL)
- raise_error ("Augeas.create");
+ raise_init_error ("Augeas.create");
CAMLreturn (Val_augeas_t (t));
}
} else if (r == 0) /* Return None */
optv = Val_int (0);
else if (r == -1) /* Error or multiple matches */
- raise_error ("Augeas.get");
+ raise_error (t, "Augeas.get");
else
failwith ("Augeas.get: bad return value");
else if (r == 0) /* Return false */
v = Val_int (0);
else if (r == -1) /* Error or multiple matches */
- raise_error ("Augeas.exists");
+ raise_error (t, "Augeas.exists");
else
failwith ("Augeas.exists: bad return value");
before = beforev == Val_int (0) ? 0 : Int_val (Field (beforev, 0));
if (aug_insert (t, path, label, before) == -1)
- raise_error ("Augeas.insert");
+ raise_error (t, "Augeas.insert");
CAMLreturn (Val_unit);
}
r = aug_rm (t, path);
if (r == -1)
- raise_error ("Augeas.rm");
+ raise_error (t, "Augeas.rm");
CAMLreturn (Val_int (r));
}
r = aug_match (t, path, &matches);
if (r == -1)
- raise_error ("Augeas.matches");
+ raise_error (t, "Augeas.matches");
/* Copy the paths to a list. */
rv = Val_int (0);
r = aug_match (t, path, NULL);
if (r == -1)
- raise_error ("Augeas.count_matches");
+ raise_error (t, "Augeas.count_matches");
CAMLreturn (Val_int (r));
}
augeas_t t = Augeas_t_val (tv);
if (aug_save (t) == -1)
- raise_error ("Augeas.save");
+ raise_error (t, "Augeas.save");
CAMLreturn (Val_unit);
}
augeas_t t = Augeas_t_val (tv);
if (aug_load (t) == -1)
- raise_error ("Augeas.load");
+ raise_error (t, "Augeas.load");
CAMLreturn (Val_unit);
}