Использование потоков stdout / stderr / stdin за FFI от haskell
Вопрос
Я разрабатываю небольшую программу на haskell, которая использует внешнюю статическую библиотеку, которую я разработал на C ++.Он обращается к библиотеке через FFI ghc (интерфейс внешней функции).Внутри этой библиотеки я хотел бы сделать некоторый вывод на консоль.Однако, это выглядит мне нравится, что сторона c ++ не имеет правильного дескриптора для стандартного вывода, потому что выходные данные не отображаются на консоли.Итак, мои вопросы таковы:
- Перехватывает ли ghc эти три потока (stdout, stdin, stderr) или libstdc ++ просто не инициализирует их, потому что я связываюсь с ghc?
- Должен ли мой импорт FFI быть "безопасным", если они записываются в стандартный вывод?
- Как я могу передать стандартный вывод в функцию C?Должен ли я просто передать его напрямую или мне нужен тип C?
Дополнительные примечания: Я связываю libstdc ++ непосредственно с исполняемым файлом (т.е.ghc -lstdc ++ ...), который, как я наивно предположил, был бы правильным способом сделать это.Кажется, работает хорошо
Отказ от ответственности: Все еще довольно новичок в Haskell, так что пока делайте маленькие шаги ; P
Решение
Похоже, ваша проблема заключается в том, что libstdc ++ не инициализируется.Я не совсем уверен, почему — -lstdc++
этого достаточно в моей системе — но посмотрите, работает ли это наоборот.
Main.hs
:
{-# LANGUAGE ForeignFunctionInterface #-}
module Main where
foreign export ccall "Main_main" main :: IO ()
foreign import ccall driver_callback :: IO ()
main = putStrLn "Now in Haskell" >> driver_callback
driver.cc
:
#include <iostream>
extern "C" {
# include "HsFFI.h"
# ifdef __GLASGOW_HASKELL__
# include "Main_stub.h"
extern void __stginit_Main(void);
# endif
void driver_callback(void) {
std::cout << "Back in C++" << std::endl;
}
}
int main(int argc, char **argv) {
hs_init(&argc, &argv);
# ifdef __GLASGOW_HASKELL__
hs_add_root(__stginit_Main);
# endif
std::cout << "Starting in C++" << std::endl;
Main_main();
hs_exit();
return 0;
}
Компиляция:
$ ghc -c --make Main [1 of 1] Compiling Main ( Main.hs, Main.o ) $ ghc --make -no-hs-main -lstdc++ Main driver.cc Linking Main ... $ ./Main Starting in C++ Now in Haskell Back in C++