Чтобы передать строку Fortran в C, также передается скрытый параметр с размером переменной. Вот рабочее определение fortran и метод C (фактически C++/CLI):
interface
subroutine AppendExtension(
+ Filename)
+ bind(C, name="AppendExtension")
character *1, intent(inout):: Filename
end subroutine AppendExtension
end interface
и вот вызываемый C++/CLI:
extern "C" {
void __declspec(dllexport) __cdecl AppendExtension(
char * name,
int buffersize)
{
String^ clistr = gcnew String(name);
clistr = System::IO::Path::ChangeExtension(clistr->Trim(), gcnew String("OUT"));
IntPtr p = Marshal::StringToHGlobalAnsi(clistr);
char *pNewCharStr = static_cast<char*>(p.ToPointer());
int cstrlen = strlen(pNewCharStr);
memcpy_s(name, buffersize, pNewCharStr, cstrlen);
if (cstrlen < buffersize)
{
// backfill with spaces, since a Fortran string is spaces on the right.
memset(&name[cstrlen], ' ', buffersize-cstrlen);
}
Marshal::FreeHGlobal(p);
}
Вышеупомянутое работает правильно (Intel Visual Fortran 2013 SP1).
Теперь я хочу передать две строки в функцию. Вот что я сделал:
interface
subroutine ParseSpec(
+ Filename, Corename)
+ bind(C, name="ParseSpec")
character *1, intent(inout):: Filename
character *1, intent(inout):: Corename
end subroutine ParseSpec
end interface
Вот призыв в действии:
CHARACTER*80 FILE_SPEC
CHARACTER*200 CORE_NAME
CALL ParseSpec(FILE_SPEC, CORE_NAME)
и вот С++/CLI:
void __declspec(dllexport) __cdecl ParseSpec(char * name, char * corename, int namelen, int corelen)
{
// namelen and corelen both contain the length of "name".
...
Есть две скрытые переменные. Я думаю, что они должны быть для размеров буфера для каждой из моих двух строк, одна для «Имя файла» и одна для «Имя ядра». Но оба содержат размер буфера первого буфера, а именно 80.
Где я ошибаюсь?