InteropServices.COMException возникает при передаче структуры c# в c ++ - PullRequest
0 голосов
/ 10 июля 2020

У меня есть Java API, к которому я хочу получить доступ через c# код, работающий на centos (Linux), работающий в версии. Net -Core 3.1.100. Следовательно, я использую sh, чтобы передать массив структур из c# в собственный код, отвечающий за запуск JVM и выполнение всего java. Моя проблема заключалась в том, что массив структур c# не попадал в код C ++ при определенных обстоятельствах. Вот код c#:

using System;
using System.Runtime.InteropServices;

namespace dotnet
{
    [StructLayout (LayoutKind.Explicit)]
    internal struct JValue {
        [FieldOffset (0)] bool z;
        [FieldOffset (0)] byte b;
        [FieldOffset (0)] char c;
        [FieldOffset (0)] short s;
        [FieldOffset (0)] int i;
        [FieldOffset (0)] long j;
        [FieldOffset (0)] float f;
        [FieldOffset (0)] double d;
        [FieldOffset (0)] IntPtr l;

        internal long JLong {
            set {
                Console.WriteLine("JLong setter set JValue as a long with " + value);
                j = value;
            }
        }

        internal float JFloat {
            set {
                Console.WriteLine("JFloat setter set JValue as a float with " + value);
                f = value;
            }
        }
    }

    class Program
    {
        [DllImport("libNative.so", CallingConvention = CallingConvention.Cdecl)]
        [return: MarshalAs(UnmanagedType.LPStr)]
        public static extern string testJvalueLong(JValue[] args, int size);

        [DllImport("libNative.so", CallingConvention = CallingConvention.Cdecl)]
        [return: MarshalAs(UnmanagedType.LPStr)]
        public static extern string testJvalueFloat(JValue[] args, int size);

        static void Main(string[] args)
        {
            JValue[] args_long = new JValue[1];
            args_long[0].JLong = 92;
            try {
                string reply = testJvalueLong(args_long, 1);
                Console.WriteLine("Long passed, native code returned: " + reply);
            }
            catch(Exception e) {
                Console.WriteLine("Error " + e.ToString());
            }

            JValue[] args_float = new JValue[1];
            args_float[0].JFloat = 18.7F;
            try {
                string reply = testJvalueFloat(args_float, 1);
                Console.WriteLine("Float passed, native code returned: " + reply);
            }
            catch(Exception e) {
                Console.WriteLine("Error " + e.ToString());
            }
        }
    }
}

Код c ++:

#include <stdio.h>
#include <jni.h>
#include <iostream>
#include <fstream>
#include <string.h>
#include <cstdlib>
#include <sstream>

typedef unsigned long ULONG;

extern "C" {
        const char * testJvalueLong(jvalue * arg, int size) {
                std::cout << "C++ testJvalueLong" << std::endl;

                std::ostringstream as_string;
                as_string << "C++ success " << arg->j;
                std::string str = as_string.str();
                const char * p = str.c_str();
                ULONG ulSize = strlen(p) + sizeof(char);

                char* pszReturn = NULL;
                pszReturn = (char*)::malloc(ulSize);
                strcpy(pszReturn, p);

                return pszReturn;
        }
        const char * testJvalueFloat(jvalue * arg, int size) {
                std::cout << "C++ testJvalueFloat" << std::endl;

                std::ostringstream as_string;
                as_string << "C++ success " << arg->f;
                std::string str = as_string.str();
                const char * p = str.c_str();
                ULONG ulSize = strlen(p) + sizeof(char);

                char* pszReturn = NULL;
                pszReturn = (char*)::malloc(ulSize);
                strcpy(pszReturn, p);

                return pszReturn;
        }
}

Когда я запускаю это, результат будет:

  JLong setter set JValue as a long with 92
  C++ testJvalueLong
  Long passed, native code returned: C++ success 92
  JFloat setter set JValue as a float with 18.7
  Error System.Runtime.InteropServices.COMException (0x8007007A): The data area passed to a system call is too small.
   (0x8007007A)
     at System.StubHelpers.MngdNativeArrayMarshaler.ConvertContentsToNative(IntPtr pMarshalState, Object& pManagedHome, IntPtr pNativeHome)
     at dotnet.Program.testJvalueFloat(JValue[] args, Int32 size)
     at dotnet.Program.Main(String[] args) in /home/vagrant/stack_exchange/dotnet/Program.cs:line 59

Однако, если я прокомментирую вне объявления элемента char в структуре c# // [FieldOffset (0)] char c;, тогда он работает, и результат будет следующим:

  JLong setter set JValue as a long with 92
  C++ testJvalueLong
  Long passed, native code returned: C++ success 92
  JFloat setter set JValue as a float with 18.7
  C++ testJvalueFloat
  Float passed, native code returned: C++ success 18.7

Любая помощь относительно того, почему это может быть, будет очень признательна.

...