Как создать таблицу (массив) с внешними значениями? - PullRequest
1 голос
/ 27 июля 2010

Я хотел бы создать статическую (область видимости) таблицу указателя данных, размера данных и версии данных. Проблема в том, что данные находятся во внешних файлах, но константы во внешних файлах.

Пример:

file1.c

    const unsigned char data1[] =
    {
    0x65, 0xF0, 0xA8, 0x5F, 0x5F,
    0x5F, 0x5F, 0x31, 0x32, 0x2E,
    0x31, 0xF1, 0x63, 0x4D, 0x43, 
    0x52, 0x45, 0x41, 0x54, 0x45,
    0x44, 0x20, 0x42, 0x59, 0x3A,
    0x20, 0x69, 0x73, 0x70, 0x56, 
// ...
    };
    const unsigned int data1_size =
        sizeof(data1) / sizeof(data1[0]);
    const unsigned int data1_version = 1;

file2.c

    const unsigned char data2[] =
    {
    0x20, 0x44, 0x61, 0x74, 0x61,
    0x20, 0x52, 0x6F, 0x77, 0x20,
    0x3D, 0x20, 0x34, 0x38, 0x12, 
//...
    };
    const unsigned int data2_size =
        sizeof(data2) / sizeof(data2[0]);
    const unsigned int data2_version = 1;

main.c

struct Data_Info
{
  unsigned char * data_ptr;
  unsigned int    data_size;
  unsigned int    data_version;
};

extern const unsigned char data1[];
extern const unsigned int  data1_size;
extern const unsigned int  data1_version;

extern const unsigned char data2[];
extern const unsigned int  data2_size;
extern const unsigned int  data2_version;

static struct Data_Info  Data_Info_Table[] =
{
  // How to set this up??
  // The compiler wants constant expressions here,
  //    and the extern is not considered a constant expression
  // This is what I tried:
  { data1, data1_size, data1_version},
  { data2, data2_size, data2_version},
};

int main(void)
{
  return 0;
} 

Я использую компилятор Green Hills, 4.2.3. Точное сообщение об ошибке:

error #28, expression must have a constant value.

см. Мой предыдущий пост:
C: Внешние константы в массиве константных структур

Примечание: тег C ++ включен, поскольку это также относится к C ++

Ответы [ 4 ]

2 голосов
/ 27 июля 2010

Вы пробовали? (где Х равно 0 или 1.)

struct Data_Info    
{    
  unsigned char const * data_ptr;    
  unsigned int    data_size;    
  unsigned int    data_version;    
};    

fileX.h

extern const struct Data_Info data_infoX;

fileX.c

const struct Data_Info data_infoX = { dataX, dataX_size, dataX_version };

main.c

const struct Data_Info const * Data_Info_Table[] = 
{ 
  &data_info1,
  &data_info2
}; 
1 голос
/ 27 июля 2010

Проблема в том, что компилятор не знает значение для размещения в data_size и data_version членах поля. Есть несколько способов обойти это без лишней суеты.

Подход 1:

#include "data1.c"
#include "data2.c"

...
static struct Data_Info  Data_Info_Table[] =
{
  { data1, sizeof(data1), 1},
  { data2, sizeof(data2), 1},
};

Подход 2:

struct Data_Info
{
  unsigned char * data_ptr;
  unsigned int *  data_size;
  unsigned int *  data_version;
};

...

static struct Data_Info  Data_Info_Table[] =
{
  { data1, &data1_size, &data1_version},
  { data2, &data2_size, &data2_version},
};

Третий подход, как уже продемонстрировал jyoung, заключается в размещении элементов в отдельных файлах данных и включении их адресов в таблицу Data_Info_Table.

0 голосов
/ 27 июля 2010

Я не такой уж эксперт в Си, но порядок инициализации в разных единицах перевода, скорее всего, не определен (я знаю, что он не определен в C ++, и это было бы огромным нарушением совместимости, если бы поведение отличалось в C ), поэтому даже если это разрешено компилятором, это будет опасно.

Теперь, добавление тега C ++, скорее всего, не очень поможет, так как AFAIK это зависит не только от языка, но и от конкретной версии языка: C89 / C99. В C ++ это работает, если вы переместите инициализацию в объявление константы в заголовочном файле. Опять же, я не эксперт по Си.

0 голосов
/ 27 июля 2010

Когда константы определены, используйте ключевое слово extern.

...