С# обмен данными с DLL C++ - C#

Узнай цену своей работы

Формулировка задачи:

Здравствуйте уважаемые ГУРУ!

Функция DLL на с++

в результате работы формирует строку типа

_bstr_t

Эту строку нужно передать в код на

С#

Делаю ледующим образом: 1. На С# получаю указатель на выделенный блок памяти и передаю его параметром в функцию dll на С++ 2. В DLL создается строка с типом _bstr_t ID, которая преобразуется в массив char и записывается по адресу переданному из С#в параметре Проблема: В С# приходит не вся строка, а только

"BaseBoard="

Само значение _bstr_t BaseBoard не передается. То есть передается только

константная часть

строки ID Если передать

только переменную_bstr_t BaseBoard

то выскакивает исключение что передаваемое значение не может быть NULL Хотя когда DLL функция тестируется как консольное приложение то все работает, Переменная _bstr_t ID содержит и константную часть и значение переменной _bstr_t BaseBoard. Помогите пожалуйста найти грабли. Заранее благодарен всем откликнувшимся . С#
Листинг программы
  1. [DllImport(@"D:\\ProjectsC++\\DLL\\MyDLL\\Debug\\MyDLL.dll",
  2. EntryPoint = "GetString",
  3. ExactSpelling = true,
  4. CallingConvention = CallingConvention.Cdecl)]
  5. public static extern int GetString(IntPtr pComputerName);
  6.  
  7. var pComputerName = Marshal.AllocHGlobal(256);
  8. // Вызываем описанную внешнюю функцию
  9. GetString(pComputerName, ref size);
  10. // Переводим результат в управляемый вид
  11. var str = Marshal.PtrToStringUni(pComputerName);
  12. //Освобождаем выделенную память
  13. Marshal.FreeHGlobal(pComputerName);
С++:
Листинг программы
  1. define _WIN32_DCOM
  2. #include <iostream>
  3. using namespace std;
  4. #include <comdef.h>#
  5. #include <Wbemidl.h>
  6. #pragma comment(lib, "wbemuuid.lib")
  7. extern "C" __declspec(dllexport) int GetString(char *pMemory)
  8. {
  9. HRESULT hres;
  10. // Step 1: --------------------------------------------------
  11. // Initialize COM. ------------------------------------------
  12. hres = CoInitializeEx(0, COINIT_MULTITHREADED);
  13. if (FAILED(hres))
  14. {
  15. cout << "Failed to initialize COM library. Error code = 0x"
  16. << hex << hres << endl;
  17. return 1; // Program has failed.
  18. }
  19. // Step 2: --------------------------------------------------
  20. // Set general COM security levels --------------------------
  21. hres = CoInitializeSecurity(
  22. NULL,
  23. -1, // COM authentication
  24. NULL, // Authentication services
  25. NULL, // Reserved
  26. RPC_C_AUTHN_LEVEL_DEFAULT, // Default authentication
  27. RPC_C_IMP_LEVEL_IMPERSONATE, // Default Impersonation
  28. NULL, // Authentication info
  29. EOAC_NONE, // Additional capabilities
  30. NULL // Reserved
  31. );
  32. if (FAILED(hres))
  33. {
  34. cout << "Failed to initialize security. Error code = 0x"
  35. << hex << hres << endl;
  36. CoUninitialize();
  37. return 1; // Program has failed.
  38. }
  39. // Step 3: ---------------------------------------------------
  40. // Obtain the initial locator to WMI -------------------------
  41. IWbemLocator *pLoc = NULL;
  42. hres = CoCreateInstance(
  43. CLSID_WbemLocator,
  44. 0,
  45. CLSCTX_INPROC_SERVER,
  46. IID_IWbemLocator, (LPVOID *) &pLoc);
  47. if (FAILED(hres))
  48. {
  49. cout << "Failed to create IWbemLocator object."
  50. << " Err code = 0x"
  51. << hex << hres << endl;
  52. CoUninitialize();
  53. return 1; // Program has failed.
  54. }
  55. // Step 4: -----------------------------------------------------
  56. // Connect to WMI through the IWbemLocator::ConnectServer method
  57. IWbemServices *pSvc = NULL;
  58. // Connect to the root\cimv2 namespace with
  59. // the current user and obtain pointer pSvc
  60. // to make IWbemServices calls.
  61. hres = pLoc->ConnectServer(
  62. _bstr_t(L"ROOT\\CIMV2"), // Object path of WMI namespace
  63. NULL, // User name. NULL = current user
  64. NULL, // User password. NULL = current
  65. 0, // Locale. NULL indicates current
  66. NULL, // Security flags.
  67. 0, // Authority (for example, Kerberos)
  68. 0, // Context object
  69. &pSvc // pointer to IWbemServices proxy
  70. );
  71. if (FAILED(hres))
  72. {
  73. cout << "Could not connect. Error code = 0x"
  74. << hex << hres << endl;
  75. pLoc->Release();
  76. CoUninitialize();
  77. return 1; // Program has failed.
  78. }
  79. cout << "Connected to ROOT\\CIMV2 WMI namespace" << endl;
  80. // Step 5: --------------------------------------------------
  81. // Set security levels on the proxy -------------------------
  82. hres = CoSetProxyBlanket(
  83. pSvc, // Indicates the proxy to set
  84. RPC_C_AUTHN_WINNT, // RPC_C_AUTHN_xxx
  85. RPC_C_AUTHZ_NONE, // RPC_C_AUTHZ_xxx
  86. NULL, // Server principal name
  87. RPC_C_AUTHN_LEVEL_CALL, // RPC_C_AUTHN_LEVEL_xxx
  88. RPC_C_IMP_LEVEL_IMPERSONATE, // RPC_C_IMP_LEVEL_xxx
  89. NULL, // client identity
  90. EOAC_NONE // proxy capabilities
  91. );
  92. if (FAILED(hres))
  93. {
  94. cout << "Could not set proxy blanket. Error code = 0x"
  95. << hex << hres << endl;
  96. pSvc->Release();
  97. pLoc->Release();
  98. CoUninitialize();
  99. return 1; // Program has failed.
  100. }
  101. // Step 6: --------------------------------------------------
  102. // Use the IWbemServices pointer to make requests of WMI ----
  103. // For example, get the name of the operating system
  104. IEnumWbemClassObject* pEnumerator = NULL;
  105. hres = pSvc->ExecQuery(
  106. bstr_t("WQL"),
  107. bstr_t("SELECT * FROM Win32_BaseBoard"),
  108. WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY,
  109. NULL,
  110. &pEnumerator);
  111. if (FAILED(hres))
  112. {
  113. cout << "Query for operating system name failed."
  114. << " Error code = 0x"
  115. << hex << hres << endl;
  116. pSvc->Release();
  117. pLoc->Release();
  118. CoUninitialize();
  119. return 1; // Program has failed.
  120. }
  121. // Step 7: -------------------------------------------------
  122. // Get the data from the query in step 6 -------------------
  123. IWbemClassObject *pclsObj = NULL;
  124. ULONG uReturn = 0;
  125. while (pEnumerator)
  126. {
  127. HRESULT hr = pEnumerator->Next(WBEM_INFINITE, 1,
  128. &pclsObj, &uReturn);
  129. if(0 == uReturn)
  130. {
  131. break;
  132. }
  133. VARIANT vtProp;
  134. // Get the value of the Name property
  135. hr = pclsObj->Get(L"SerialNumber", 0, &vtProp, 0, 0); //BSTR vtProp
  136. _bstr_t BaseBoard(vtProp.bstrVal);
  137. _bstr_t ID = "BaseBoard=" + BaseBoard;
  138. [B]//преобразуем _bstr_t ID в массив char и записываем его по переданному в параметре адресу[/B]
  139. strcpy_s(pMemory, ID.length() + 1, (LPCSTR)ID);
  140. VariantClear(&vtProp);
  141. pclsObj->Release();
  142. }
  143. // Cleanup
  144. pSvc->Release();
  145. pLoc->Release();
  146. pEnumerator->Release();
  147. CoUninitialize();
  148. return 0;
  149. }

Решение задачи: «С# обмен данными с DLL C++»

textual
Листинг программы
  1. string wmiQuery = "SELECT * FROM Win32_OperatingSystem";
  2.  
  3. using(ManagementObjectSearcher searcher = new ManagementObjectSearcher ( wmiQuery ))
  4. {
  5.    ManagementObjectCollection retObjectCollection = searcher.Get ( );
  6.  
  7.    foreach (ManagementObject retObject in retObjectCollection)
  8.    {
  9.        string result = retObject["Name"];
  10.        break;
  11.    }
  12. }

ИИ поможет Вам:


  • решить любую задачу по программированию
  • объяснить код
  • расставить комментарии в коде
  • и т.д
Попробуйте бесплатно

Оцени полезность:

15   голосов , оценка 4.067 из 5

Нужна аналогичная работа?

Оформи быстрый заказ и узнай стоимость

Бесплатно
Оформите заказ и авторы начнут откликаться уже через 10 минут