summaryrefslogtreecommitdiffstats
path: root/generator
diff options
context:
space:
mode:
authorRichard W.M. Jones <rjones@redhat.com>2011-03-28 10:03:16 +0100
committerRichard W.M. Jones <rjones@redhat.com>2011-03-28 10:03:16 +0100
commite751293e10d5ecbb2ef43a61b9c153a1fc4f0304 (patch)
tree4c1049f6ab884bff5d850004bcbd427cfaf85f13 /generator
parenta0e3b2183733c7440ad017c4ffe361935672339b (diff)
downloadlibguestfs-e751293e10d5ecbb2ef43a61b9c153a1fc4f0304.tar.gz
libguestfs-e751293e10d5ecbb2ef43a61b9c153a1fc4f0304.tar.xz
libguestfs-e751293e10d5ecbb2ef43a61b9c153a1fc4f0304.zip
ruby: Don't segfault if callbacks throw exceptions (RHBZ#664558).
(Thanks Chris Lalancette). See: https://bugzilla.redhat.com/show_bug.cgi?id=664558#c6
Notes
Notes: Labels: bugfix, RHBZ#664558 Depends: 6a64114929a0b098f5a1e31e17e7802127925007
Diffstat (limited to 'generator')
-rw-r--r--generator/generator_ruby.ml49
1 files changed, 44 insertions, 5 deletions
diff --git a/generator/generator_ruby.ml b/generator/generator_ruby.ml
index 84d10e00..7c8788d0 100644
--- a/generator/generator_ruby.ml
+++ b/generator/generator_ruby.ml
@@ -55,6 +55,8 @@ static VALUE c_guestfs; /* guestfs_h handle */
static VALUE e_Error; /* used for all errors */
static void ruby_event_callback_wrapper (guestfs_h *g, void *data, uint64_t event, int event_handle, int flags, const char *buf, size_t buf_len, const uint64_t *array, size_t array_len);
+static VALUE ruby_event_callback_wrapper_wrapper (VALUE argv);
+static VALUE ruby_event_callback_handle_exception (VALUE not_used, VALUE exn);
static VALUE **get_all_event_callbacks (guestfs_h *g, size_t *len_rtn);
static void
@@ -212,7 +214,7 @@ ruby_event_callback_wrapper (guestfs_h *g,
const uint64_t *array, size_t array_len)
{
size_t i;
- VALUE eventv, event_handlev, bufv, arrayv;
+ VALUE eventv, event_handlev, bufv, arrayv, argv;
eventv = ULL2NUM (event);
event_handlev = INT2NUM (event_handle);
@@ -223,12 +225,49 @@ ruby_event_callback_wrapper (guestfs_h *g,
for (i = 0; i < array_len; ++i)
rb_ary_push (arrayv, ULL2NUM (array[i]));
- /* XXX If the Ruby callback raises any sort of exception then
- * it causes the process to segfault. I don't understand how
- * to catch exceptions here.
+ /* Wrap up the arguments in an array which will be unpacked
+ * and passed as multiple arguments. This is a crap limitation
+ * of rb_rescue.
+ * http://blade.nagaokaut.ac.jp/cgi-bin/scat.rb/~poffice/mail/ruby-talk/65698
*/
- rb_funcall (*(VALUE *) data, rb_intern (\"call\"), 4,
+ argv = rb_ary_new2 (5);
+ rb_ary_store (argv, 0, * (VALUE *) data /* function */);
+ rb_ary_store (argv, 1, eventv);
+ rb_ary_store (argv, 2, event_handlev);
+ rb_ary_store (argv, 3, bufv);
+ rb_ary_store (argv, 4, arrayv);
+
+ rb_rescue (ruby_event_callback_wrapper_wrapper, argv,
+ ruby_event_callback_handle_exception, Qnil);
+}
+
+static VALUE
+ruby_event_callback_wrapper_wrapper (VALUE argv)
+{
+ VALUE fn, eventv, event_handlev, bufv, arrayv;
+
+ fn = rb_ary_entry (argv, 0);
+ eventv = rb_ary_entry (argv, 1);
+ event_handlev = rb_ary_entry (argv, 2);
+ bufv = rb_ary_entry (argv, 3);
+ arrayv = rb_ary_entry (argv, 4);
+
+ rb_funcall (fn, rb_intern (\"call\"), 4,
eventv, event_handlev, bufv, arrayv);
+
+ return Qnil;
+}
+
+static VALUE
+ruby_event_callback_handle_exception (VALUE not_used, VALUE exn)
+{
+ /* Callbacks aren't supposed to throw exceptions. The best we
+ * can do is to print the error.
+ */
+ fprintf (stderr, \"libguestfs: exception in callback: %%s\",
+ StringValueCStr (exn));
+
+ return Qnil;
}
static VALUE **