AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Thema durchsuchen
Ansicht
Themen-Optionen

Delphi C++ Dll

Ein Thema von mandoza · begonnen am 27. Jan 2019 · letzter Beitrag vom 28. Jan 2019
Antwort Antwort
mandoza

Registriert seit: 27. Apr 2018
16 Beiträge
 
#1

Delphi C++ Dll

  Alt 27. Jan 2019, 22:57
Delphi-Version: 10.2 Tokyo
Hello every one ; I've a C++ DLL that i want to consume from Delphi Project ; My C++ part is

Delphi-Quellcode:
//Header file UserIdentity.h
extern "C" {
#endif
struct UserIdentity
{
    const char* name;
    int id;
}
;
/**
 * Gets all of the UserIdentity objects currently available.
 * @param[out] pCount A pointer to storage for the number of UserIdentity objects available
 * @param[out] pUserIdentities A pointer to storage for a number of UserIdentity objects indicated by pCount
 * \n NOTE : Pass nullptr for pUserIdentities to get only the count value
 * @usage
 *
 * size_t userIdentitiesCount = 0;
 * UserIdentity* userIdentities = nullptr;
 * Ui_export(&userIdentitiesCount, nullptr);
 * if (userIdentitiesCount) {
*      userIdentities = new UserIdentity[userIdentitiesCount];
*      Ui_export(&userIdentitiesCount, userIdentities);
*      // Process userIdentities...
*      delete [] userIdentities;
*  }

 *
 */
EXAMPLE_LIBRARY_EXPORT void Ui_export(size_t* pCount, UserIdentity* pUserIdentities);
#ifdef __cplusplus
}
#endif
And here's my cpp fie " UserIdentity.cpp "

Delphi-Quellcode:
#include "UserIdentity.h"
#include <map>
static const std::map<std::string, int> gUserIdentites = {
    { "Bob", 100 }
,
    { "Jone", 101 },
    { "Alice", 102 },
    { "Doe", 103 }
};
void Ui_export(size_t* pCount, UserIdentity* pUserIdentities)
{
    // NOTE : If pCount is a valid pointer then we'll set pCount...
    if (pCount) {
        *pCount = gUserIdentites.size();
    }

    // NOTE : If pUserIdentities is a valid pointer then we'll populate it with
    // data...it's the resposibility of the caller to ensure they've correctly
    // gotten the count and have allocated enough storage for the data...
    if (pUserIdentities) {
        size_t i = 0;
        for (const auto& userIdentity : gUserIdentites) {
            // NOTE : Becuase we're getting a C string using the key's c_str()
            //  method we need to make sure that we document cases which will
            //  cause map elements to become invalidated.
            //  The API could be made more robust by having allocators for
            //  dynamic strings, but this would introduce complexity that may
            //  not be necessary.
            pUserIdentities[i].name = userIdentity.first.c_str();
            pUserIdentities[i].id = userIdentity.second;
            ++i;
        }

    }
}
Finally my Delphi attempt part :
Delphi-Quellcode:
Type
UserIdentity = record
    name_ : PAnsiChar;
    id : Integer;
  end;
  PUserIdentity = ^UserIdentity;
procedure Ui_export(pCount : PUInt32; pUserIdentities : PUserIdentity);cdecl; external Cdllname;
....
procedure TForm39.Button24Click(Sender: TObject);
var
  pCount : UInt32;
  LUserEntity : PUserIdentity;
  LUserEntityData : UserIdentity;
  i: Integer;
begin
  Ui_export(@pCount, nil);
  GetMem(LUserEntity, pCount);
  Ui_export(@pCount, LUserEntity);
  for i := 0 to pCount - 1 do
  begin
    memo1.Lines.Add(LUserEntity^.name_+' '+LUserEntity^.id.ToString);
    Inc(LUserEntity);
  end;
end;
The issue with delphi is that i get the 2 first values correctly that's :
Bob
Jone

But for
Alice i get some rubbishes value which is ' yyyyyyyyyyyyyyyyyyyyyyy'

for
Doe
i get it's correct value .

So please is there any incorrect part in my Delphi Implementation ?

Geändert von mandoza (27. Jan 2019 um 23:02 Uhr)
  Mit Zitat antworten Zitat
Klaus01
Online

Registriert seit: 30. Nov 2005
Ort: München
5.754 Beiträge
 
Delphi 10.4 Sydney
 
#2

AW: Delphi C++ Dll

  Alt 28. Jan 2019, 07:11
Hi,

will it make a difference when using a packet record instead the unpacked one?

Delphi-Quellcode:
UserIdentity = packet record
    name_ : PAnsiChar;
    id : Integer;
  end;
Best regards
Klaus
Klaus
  Mit Zitat antworten Zitat
Benutzerbild von Zacherl
Zacherl

Registriert seit: 3. Sep 2004
4.629 Beiträge
 
Delphi 10.2 Tokyo Starter
 
#3

AW: Delphi C++ Dll

  Alt 28. Jan 2019, 08:22
size_t is a plaform and architecture specific type ... using UInt32 might not be correct e.g. for 64-bit. The real problem is this part tho: GetMem(LUserEntity, pCount); It has to be GetMem(LUserEntity, pCount * SizeOf(UserEntity));
Projekte:
- GitHub (Profil, zyantific)
- zYan Disassembler Engine ( Zydis Online, Zydis GitHub)

Geändert von Zacherl (28. Jan 2019 um 08:26 Uhr)
  Mit Zitat antworten Zitat
Der schöne Günther
Online

Registriert seit: 6. Mär 2013
6.110 Beiträge
 
Delphi 10 Seattle Enterprise
 
#4

AW: Delphi C++ Dll

  Alt 28. Jan 2019, 08:36
Why even multiply with `count`? The size of the struct is always the same. It only contains a pointer, not the characters themselves.
  Mit Zitat antworten Zitat
Benutzerbild von Zacherl
Zacherl

Registriert seit: 3. Sep 2004
4.629 Beiträge
 
Delphi 10.2 Tokyo Starter
 
#5

AW: Delphi C++ Dll

  Alt 28. Jan 2019, 08:48
Why even multiply with `count`? The size of the struct is always the same. It only contains a pointer, not the characters themselves.
Nope. It fills the memory described by the pointer with the list elements. In C/C++ int* is basically the same as int[] . The struct size is ofc. the same, but the number of elements in the array might be different each time, so you have to alloc enough space to prevent buffer overrides (thats what happening without the *count).
Projekte:
- GitHub (Profil, zyantific)
- zYan Disassembler Engine ( Zydis Online, Zydis GitHub)
  Mit Zitat antworten Zitat
Der schöne Günther
Online

Registriert seit: 6. Mär 2013
6.110 Beiträge
 
Delphi 10 Seattle Enterprise
 
#6

AW: Delphi C++ Dll

  Alt 28. Jan 2019, 09:19
My bad, I did not read properly. I thought "count" was referring to the number of characters in the string, not the number of structs returned.
  Mit Zitat antworten Zitat
mandoza

Registriert seit: 27. Apr 2018
16 Beiträge
 
#7

AW: Delphi C++ Dll

  Alt 28. Jan 2019, 10:28
size_t is a plaform and architecture specific type ... using UInt32 might not be correct e.g. for 64-bit. The real problem is this part tho: GetMem(LUserEntity, pCount); It has to be GetMem(LUserEntity, pCount * SizeOf(UserEntity));
Thank you Zacherl , I've fixed the issue by using string fixed length in both C++ and Delphi , and using the UIntPtr instead of UInt32

Again thank you .
  Mit Zitat antworten Zitat
Antwort Antwort


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 09:43 Uhr.
Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz