Как вы используете Ruby/DL? Это правильно?

Я пытаюсь написать интерфейс между RSPEC (BDD со вкусом рубина) и приложением Windows. Само приложение написано на неясном языке, но оно имеет C API для обеспечения доступа. Я перешел на Ruby/DL, но у меня возникают трудности с выполнением даже самого простого вызова метода DLL. Вот что у меня есть в файле с именем gt4r.rb:

require 'dl/import'

module Gt4r
  extend DL::Importable
  dlload 'c:\\gtdev\\r321\\bin\\gtvapi'

  # GTD initialization/termination functions
  extern 'int GTD_init(char *[], char *, char *)'
  extern 'int GTD_initialize(char *, char *, char *)'
  extern 'int GTD_done(void)'
  extern 'int GTD_get_error_message(int, char **)'
end

Мое чтение пока говорит о том, что это все, что мне нужно для начала, поэтому я написал пример RSPEC:

require 'gt4r'

@@test_environment = "INCLUDE=C:\\graphtalk\\env\\aiadev\\config\\aiadev.ini"
@@normal_user = "BMCHARGUE"

describe Gt4r do
  it 'initializes' do
      rv = Gt4r.gTD_initialize @@normal_user, @@normal_user, @@test_environment
      rv.should == 0
  end
end

И когда беги...

C:\code\GraphTalk>spec -fs -rgt4r gt4r_spec.rb

Gt4r
- initializes (FAILED - 1)

1)
'Gt4r initializes' FAILED
expected: 0,
     got: 13 (using ==)
./gt4r_spec.rb:9:

Finished in 0.031 seconds

1 example, 1 failure

Возвращаемое значение (13) является действительным кодом возврата, что означает ошибку, но когда я пытаюсь добавить вызов gTD_get_error_message к моему RSPEC, я не могу заставить параметры работать.

Я иду в правильном направлении, и может ли кто-нибудь указать на следующую вещь, которую я могу попробовать?

Спасибо бретт


Продолжение этого вопроса, показывающее, что происходит сбой при попытке получить сообщение об ошибке из моей целевой библиотеки:

require 'gt4r'

@@test_environment = "INCLUDE=C:\\graphtalk\\env\\aiadev\\config\\aiadev.ini"
@@normal_user = "BMCHARGUE"

describe Gt4r do
  it 'initializes' do
      rv = Gt4r.gTD_initialize @@normal_user, @@normal_user, @@test_environment
      Gt4r.gTD_get_error_message rv, @msg
      @msg.should == ""
      rv.should == 0
  end
end

Я ожидаю, что сообщение об ошибке будет возвращено в @msg, но при запуске я получаю следующее:

Gt4r
(eval):5: [BUG] Segmentation fault
ruby 1.8.6 (2008-08-11) [i386-mswin32]


This application has requested the Runtime to terminate it in an unusual way.
Please contact the application's support team for more information.

И это, если я использую символ (: MSG) вместо:

C:\code\GraphTalk\gt4r_dl>spec -fs -rgt4r gt4r_spec.rb

Gt4r
- initializes (ERROR - 1)

1)
NoMethodError in 'Gt4r initializes'
undefined method `to_ptr' for :msg:Symbol
(eval):5:in `call'
(eval):5:in `gTD_get_error_message'
./gt4r_spec.rb:9:

Finished in 0.046 seconds

1 example, 1 failure

Я явно что-то упускаю из-за передачи параметров между ruby ​​и C, но что?

3 ответа

Решение

Общее мнение заключается в том, что вы хотите максимально избегать DL. (Английская) документация довольно схематична, и интерфейс трудно использовать для чего-либо, кроме тривиальных примеров.

Родной C-интерфейс на Ruby НАМНОГО проще программировать. Или вы могли бы использовать FFI, которая заполняет аналогичную нишу для DL, изначально пришла из проекта rubinius и недавно была портирована на "нормальный" рубин. Он имеет более приятный интерфейс и гораздо менее болезненный в использовании:

http://blog.headius.com/2008/10/ffi-for-ruby-now-available.html

Возвращаемое значение (13) является действительным кодом возврата, что означает ошибку, но когда я пытаюсь добавить вызов gTD_get_error_message к моему RSPEC, я не могу заставить параметры работать.

Это может помочь опубликовать ошибку вместо кода, который работал:)

По сути, как только вы начинаете иметь дело с указателями, как в (int, char **), все становится ужасно.

Вам нужно выделить указатель данных для msg для записи, так как в противном случае C некуда будет писать сообщения об ошибках. Используйте DL.mallo.

Другие вопросы по тегам