Forum: Ruby C extensions question

Announcement (2017-05-07): www.ruby-forum.com is now read-only since I unfortunately do not have the time to support and maintain the forum any more. Please see rubyonrails.org/community and ruby-lang.org/en/community for other Rails- und Ruby-related community platforms.
4d4de7748b0f480a1ae4295bf269a0ce?d=identicon&s=25 Jared Richardson (Guest)
on 2006-03-23 04:04
(Received via mailing list)
Hi all,

I haven't done any "real" work in C in nearly 10 years and I'm still
learning Ruby, so it's quite possible that I'm just doing something
dumb,
but I'm stuck here. I'm trying to call Ingres database code from Ruby,
so
I'm writing some C extensions.

I have Ruby calling into C code. Very easy and very nice.

I have Ingres sample code (which is C) compiling and running fine.

However, when I try to invoke Ingres code from any of my C code and run
it
(from Ruby), I get a seg fault.

It feels like Ruby can't see the Ingres libraries but the code samples
don't
seem to having any problems. I suspect I'm missing something simple...

Do you need to do any special wrapping or declaring of code that isn't
touched from within your Ruby? I've been through the Pickaxe book and
several online samples/tutorials and I don't see that it's needed... but
then again, my code isn't running. :)

Here's my code as it is now... any hints or clues?

#include "ruby.h"
# include <stdio.h>
# include <iiapi.h>

static VALUE cIngres;

static VALUE ii_exec(VALUE self) {
 printf("\n\nExec was run from within the Ruby code \n\n");
  IIAPI_INITPARM  initParm;
  printf("one\n");
  IIapi_initialize( &initParm );
  printf("two\n");
 return Qnil;
}

void Init_Ingres() {
 cIngres = rb_define_class("Ingres", rb_cObject);
 rb_define_method(cIngres, "exec", ii_exec, 0);
}


Thanks!

Jared
http://jaredrichardson.net
57050802d7220041a073ddb782b5cd38?d=identicon&s=25 Zeng Ke (Guest)
on 2006-03-23 04:18
(Received via mailing list)
You'd better make the Makefile using extconf.rb, in which you can
specify the library path and include path,  as is told in the rubybook.
4d4de7748b0f480a1ae4295bf269a0ce?d=identicon&s=25 Jared Richardson (Guest)
on 2006-03-23 05:34
(Received via mailing list)
Good point. I should've mentioned that I am already doing that.


"Zeng Ke" <zengke@ce-lab.net> wrote in message
news:442214D4.7050602@ce-lab.net...
5a601582df3b42b65a5e8353fc9305da?d=identicon&s=25 Gerardo Santana Gómez Garrido (Guest)
on 2006-03-23 07:35
(Received via mailing list)
Hello Jared.

I'm not familiar with Ingres, but according to

http://www.ingres.com/download/oapi.pdf

and this

http://www.rubycentral.com/book/ext_ruby.html

I'd write it this way:

-------------- >8 --------------
#include "ruby.h"
#include <iiapi.h>

static VALUE rb_cIngres;

static void
ingres_free(void *p)
{
	IIAPI_RELENVPARM relEnvParm;
	IIAPI_TERMPARM termParm;

	relEnvParm.re_envHandle = p;
	IIapi_releaseEnv(&relEnvParm);
	/* test return status */
	IIapi_terminate(&termParm);
	/* test return status */
}

static VALUE
ingres_alloc(VALUE klass)
{
	IIAPI_INITPARM  initParm;
	/*
	 * initialize initParm structure here
	 * before passing it along to IIapi_initialize
	 */
	IIapi_initialize(&initParm);
	if (initParm.in_status != IIAPI_ST_SUCCESS) {
		initParm.in_envHandle = NULL;
		/* do something else probably */
	}
	return Data_Wrap_Struct(klass, 0, ingres_free, initParm.in_envHandle);
}

static VALUE
ingres_init(VALUE self)
{
	/* Initialization stuff */
	return self;
}

void Init_ingres(void) {
	rb_cIngres = rb_define_class("Ingres", rb_cObject);
	rb_define_method(rb_cIngres, "initialize", ingres_init, 0);
}
-------------- 8< --------------


But I can't test it.

2006/3/22, Jared Richardson <jaredRMV-richaTHSrdson@nc.rr.com>:
>
>
>   IIAPI_INITPARM  initParm;
>
>
> Thanks!
>
> Jared
> http://jaredrichardson.net
>
>
>
>


--
Gerardo Santana
"Between individuals, as between nations, respect for the rights of
others is peace" - Don Benito Juárez
http://santanatechnotes.blogspot.com/
4d4de7748b0f480a1ae4295bf269a0ce?d=identicon&s=25 Jared Richardson (Guest)
on 2006-03-23 13:49
(Received via mailing list)
My original code was written just like that but I cut it down for the
newsgroup posting. :)

Thanks!


"Gerardo Santana Gómez Garrido" <gerardo.santana@gmail.com> wrote in
message
news:b06e22130603222235r5db97687l@mail.gmail.com...
Hello Jared.

I'm not familiar with Ingres, but according to

http://www.ingres.com/download/oapi.pdf

and this

http://www.rubycentral.com/book/ext_ruby.html

I'd write it this way:

-------------- >8 --------------
#include "ruby.h"
#include <iiapi.h>

static VALUE rb_cIngres;

static void
ingres_free(void *p)
{
IIAPI_RELENVPARM relEnvParm;
IIAPI_TERMPARM termParm;

relEnvParm.re_envHandle = p;
IIapi_releaseEnv(&relEnvParm);
/* test return status */
IIapi_terminate(&termParm);
/* test return status */
}

static VALUE
ingres_alloc(VALUE klass)
{
IIAPI_INITPARM  initParm;
/*
* initialize initParm structure here
* before passing it along to IIapi_initialize
*/
IIapi_initialize(&initParm);
if (initParm.in_status != IIAPI_ST_SUCCESS) {
initParm.in_envHandle = NULL;
/* do something else probably */
}
return Data_Wrap_Struct(klass, 0, ingres_free, initParm.in_envHandle);
}

static VALUE
ingres_init(VALUE self)
{
/* Initialization stuff */
return self;
}

void Init_ingres(void) {
rb_cIngres = rb_define_class("Ingres", rb_cObject);
rb_define_method(rb_cIngres, "initialize", ingres_init, 0);
}
-------------- 8< --------------


But I can't test it.

2006/3/22, Jared Richardson <jaredRMV-richaTHSrdson@nc.rr.com>:
>
> then again, my code isn't running. :)
>  printf("\n\nExec was run from within the Ruby code \n\n");
> }
>
>
> Thanks!
>
> Jared
> http://jaredrichardson.net
>
>
>
>


--
Gerardo Santana
"Between individuals, as between nations, respect for the rights of
others is peace" - Don Benito Juárez
http://santanatechnotes.blogspot.com/
4d4de7748b0f480a1ae4295bf269a0ce?d=identicon&s=25 Jared Richardson (Guest)
on 2006-03-25 02:49
(Received via mailing list)
I finally got this thing working so I thought I'd share the answer, even
though it was a simple problem (once I found it.) :)

I was already generating my Makefile with extconf.rb but I hadn't gotten
it
to answer "yes" to any of the have_library or find_libary calls. I did
have
it working for have_header. I should've focused on that have_library
that
answered "no", but I thought that since the original C samples could run
with the libraries in the LD_LIBRARY_PATH environment variable that the
Ruby
code would be okay. It wasn't. :)

It turns out that I wasn't using the right library name. The include was
called "iiapi.h" and the compiled library was named libiiapi.a (static)
and
libiiapi.1.so (dynamic). However the name of the library seems to be
"ingres". I'm pretty sure I tried using "ingres" early in the debugging
cycle, but it appears that either I didn't or that I had the environment
messed up at the time.

I haven't done anything "real" with the resulting code yet, but I can
call
into the ingres library without a segmentation fault, so I'm going to
call
it good and go take a break. :)

I'm pasting in my extconf.rb for future reference.

-----
require 'mkmf'

dir_config("ingres")
have_header("iiapi.h")
have_library("ingres")

$CFLAGS = " -g -Dint_lnx -O "
incdir = ENV["II_SYSTEM"] + "/ingres/files"
$CFLAGS += "-I#{incdir}"
libdir = ENV["II_SYSTEM"] + "/ingres/lib"
$LDFLAGS += "-L#{libdir}"
$LDFLAGS += " -lingres -lpthread -lm -lc -lcrypt -ldl -lgcc_s"

create_makefile("Ingres")
-----

Thanks for the suggestions and help.

Jared
http://jaredrichardson.net


--------------------------
--- Original Message ----
--------------------------


Hi all,

I haven't done any "real" work in C in nearly 10 years and I'm still
learning Ruby, so it's quite possible that I'm just doing something
dumb,
but I'm stuck here. I'm trying to call Ingres database code from Ruby,
so
I'm writing some C extensions.

I have Ruby calling into C code. Very easy and very nice.

I have Ingres sample code (which is C) compiling and running fine.

However, when I try to invoke Ingres code from any of my C code and run
it
(from Ruby), I get a seg fault.

It feels like Ruby can't see the Ingres libraries but the code samples
don't
seem to having any problems. I suspect I'm missing something simple...

Do you need to do any special wrapping or declaring of code that isn't
touched from within your Ruby? I've been through the Pickaxe book and
several online samples/tutorials and I don't see that it's needed... but
then again, my code isn't running. :)

Here's my code as it is now... any hints or clues?

#include "ruby.h"
# include <stdio.h>
# include <iiapi.h>

static VALUE cIngres;

static VALUE ii_exec(VALUE self) {
 printf("\n\nExec was run from within the Ruby code \n\n");
  IIAPI_INITPARM  initParm;
  printf("one\n");
  IIapi_initialize( &initParm );
  printf("two\n");
 return Qnil;
}

void Init_Ingres() {
 cIngres = rb_define_class("Ingres", rb_cObject);
 rb_define_method(cIngres, "exec", ii_exec, 0);
}


Thanks!

Jared
http://jaredrichardson.net
4d4de7748b0f480a1ae4295bf269a0ce?d=identicon&s=25 Jared Richardson (Guest)
on 2006-03-25 02:59
(Received via mailing list)
Also, I forgot to add, in order for this to work, invoke with these
options:

ruby -r mkmf
xtconf.rb  --with-ingres-include="/opt/Ingres/IngresII/ingres/files/"
--with-ingres-lib="/opt/Ingres/IngresII/ingres/lib/"


"Jared Richardson" <jaredRMV-richaTHSrdson@nc.rr.com> wrote in message
news:Yi1Vf.44009$915.14388@southeast.rr.com...
This topic is locked and can not be replied to.