Добавление указателя на типирование в Delphi

0

Я пытаюсь понять, как я могу придать тип в Delphi, как и C/C++.

Код C++, который я пытаюсь преобразовать, - это;

typedef struct tagEXT2_INODE
{
    uint16_t    i_mode;     /* File mode */
    uint16_t    i_uid;      /* Low 16 bits of Owner Uid */
    uint32_t    i_size;     /* Size in bytes */
    uint32_t    i_atime;        /* Access time */
    uint32_t    i_ctime;        /* Creation time */
    uint32_t    i_mtime;        /* Modification time */
    uint32_t    i_dtime;        /* Deletion Time */
    uint16_t    i_gid;      /* Low 16 bits of Group Id */
    uint16_t    i_links_count;  /* Links count */
    uint32_t    i_blocks;       /* Blocks count */
    uint32_t    i_flags;        /* File flags */
    union {
        struct {
            uint32_t  l_i_reserved1;
        } linux1;
        struct {
            uint32_t  h_i_translator;
        } hurd1;
        struct {
            uint32_t  m_i_reserved1;
        } masix1;
    } osd1;             /* OS dependent 1 */
    uint32_t    i_block[EXT2_N_BLOCKS];/* Pointers to blocks */
    uint32_t    i_generation;   /* File version (for NFS) */
    uint32_t    i_file_acl;     /* File ACL */
//    uint32_t  i_dir_acl;      /* Directory ACL */
    uint32_t    i_size_high;            /* This is used store the high 32 bit of file size in large files */
    uint32_t    i_faddr;        /* Fragment address */
    union {
        struct {
            uint8_t l_i_frag;   /* Fragment number */
            uint8_t l_i_fsize;  /* Fragment size */
            uint16_t    i_pad1;
            uint16_t    l_i_uid_high;   /* these 2 fields    */
            uint16_t    l_i_gid_high;   /* were reserved2[0] */
            uint32_t    l_i_reserved2;
        } linux2;
        struct {
            uint8_t h_i_frag;   /* Fragment number */
            uint8_t h_i_fsize;  /* Fragment size */
            uint16_t    h_i_mode_high;
            uint16_t    h_i_uid_high;
            uint16_t    h_i_gid_high;
            uint16_t    h_i_author;
        } hurd2;
        struct {
            uint8_t m_i_frag;   /* Fragment number */
            uint8_t m_i_fsize;  /* Fragment size */
            uint16_t    m_pad1;
            uint32_t    m_i_reserved2[2];
        } masix2;
    } osd2;                 /* OS dependent 2 */
} __attribute__ ((__packed__)) EXT2_INODE;


EXT2_INODE *src;
char *inode_buffer;
int inode_index, ret = 0;

inode_buffer = (char *)malloc(4096);


src = (EXT2_INODE *)(inode_buffer + inode_index);

Может ли кто-нибудь объяснить, как этот src может быть подобным образом? И как бы я сделал ту же операцию в Delphi?

благодаря

Изменение: вот моя запись Ext2 iNode; Тип

  TExt2iNode = Record
    i_mode : Word;
    i_uid : Word;
    i_size : Cardinal;
    i_atime : Cardinal;
    i_ctime : Cardinal;
    i_mtime : Cardinal;
    i_dtimes : Cardinal;
    i_gid : Word;
    i_links_count : Word;
    i_blocks: Cardinal;
    i_flags : Cardinal;
    osd1 : Record
      linux1 : Record
        l_i_reserved1 : Cardinal;
      end;
      hurd1 : Record
        h_i_translator: Cardinal;
      End;
      masix1 : Record
        m_i_reserved1 : Cardinal;
      End;
    End;
    i_block: array [0..EXT2_N_BLOCKS-1] of Cardinal;
    i_generation : Cardinal;
    i_file_acl : Cardinal;
    i_size_high : Cardinal;
    i_faddr : Cardinal;

    osd2 : Record
      Linux2 : Record
        l_i_frag : Byte;
        l_i_fsize : Byte;
        i_pad1 : Word;
        l_i_uid_high : Word;
        l_i_gid_high : Word;
        l_i_reserved2 : Cardinal
      end;
      hurd2 : Record
        h_i_frag : Byte;
        h_i_fsize : Byte;
        h_i_mode_high : Word;
        h_i_uid_high : Word;
        h_i_gid_high : Word;
        h_i_author : Word;
      end;
      masix2 : Record
        m_i_frag:Byte;
        m_i_fsize : Byte;
        m_pad1 : Word;
        m_i_reserved : array[0..1] of Cardinal;
      end;
    end;
  End;

Вот моя запись раздела Ex2;

type 
  Ext2Partition = Class 
    private
      handle: THandle;
      sect_size: Integer;
      total_sectors: Int64;
      relative_sect: Int64;
      linux_name :AnsiString;

      inodes_per_group: integer;
      inode_size: integer;
      block_size: integer;
      totalGRoups: Integer;
      desc : TExt2_Group_Desc;

      last_block : Cardinal;
      inode_buffer : array of AnsiChar;
      root : Ext2File;
      buffercache : TList;
      lvol : LogicalVolume;
    public
      onview, is_valid: boolean;
    published
      Constructor Create(size, offset :int64; ssise: integer; PHandle: THandle);
      function read_inode(inum: Cardinal):Ext2File;
      function readblock(blocknum: cardinal; var buffer: array of AnsiChar):integer;
      function mount():integer;
  End;

Вот моя функция read_inode где выполняется расчет указателя;

function Ext2Partition.read_inode(inum: Cardinal):Ext2File;
var
  group, index, blknum: cardinal;
  inode_index : integer;
  ret: integer;
  fFile: EXt2File;
  src: TExt2iNode;
begin
  if inum = 0 then
    Result := NIL;

  SetLength(self.inode_buffer, self.block_size);

  group := (inum -1) DIV self.inodes_per_group;

  if group > self.totalGRoups then
  begin
    //ShowMessage('Error reading iNode');
    Result := -1;
  end;

  index := ((inum-1) MOD self.inodes_per_group) * self.inode_size;
  inode_index := (index MOD self.block_size);
  blknum := self.desc.bg_inode_table + (index div self.block_size);

  if blknum <> self.last_block then
    ret := readblock(blknum, self.inode_buffer);

  fFile := TExt2iNode.Create;

  //What goes here?

end;
  • 0
    Это не типотипирование как класс . Это типизация содержимого памяти после увеличения адреса указателя на struct (запись в Delphi).
Теги:

1 ответ

1
Лучший ответ

Код выделяет блок памяти. В другом месте он определил смещение структуры inode ext2 где-то внутри этого блока, заданного inode_index. Он добавляет смещение к началу блока, указывая адрес структуры. Затем код вводит результат, чтобы сообщить компилятору, что вычисленный адрес char* действительно является адресом этого типа структуры.

Используя буквальный перевод, у нас будут следующие объявления Delphi:

var
  inode_buffer: PAnsiChar;
  inode_index: Integer;
  src: PExt2_Inode;

Поэтому, чтобы назначить src, вы должны вводить и добавлять текст точно так же, как в коде C++:

src := PExt2_Inode(inode_buffer + inode_index);

Большинство типов указателей Delphi по умолчанию не поддерживают такую арифметику указателя, но PAnsiChar является особенным.

Используя ваш перевод до сих пор, где inode_buffer - это массив вместо указателя на блок памяти, вы должны иметь следующее:

src := PExt2_Inode(@inode_buffer[inode_index]);

Это индексирует массив, а затем использует оператор @ для получения адреса этого элемента массива. (Фактически, вы могли бы использовать тот же синтаксис, если inode_buffer был моим исходным типом PAnsiChar.)

  • 0
    Спасибо за ваш вклад. Я добавил свою запись раздела Ext2, где inode_buffer . Я использовал array of AnsiChar .
  • 0
    Есть ли польза от использования указателя, а не array of ansichar ? Также пробуя ваше предложение, я получаю ошибку компиляции - Invalid Typecast .
Показать ещё 6 комментариев

Ещё вопросы

Сообщество Overcoder
Наверх
Меню