Как вы используете 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.