Использование потоков stdout / stderr / stdin за FFI от haskell

StackOverflow https://stackoverflow.com/questions/1917890

  •  20-09-2019
  •  | 
  •  

Вопрос

Я разрабатываю небольшую программу на 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++
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top