【问题标题】:expression must be a pointer to a complete object type from macro表达式必须是指向宏中完整对象类型的指针
【发布时间】:2013-08-18 17:58:21
【问题描述】:

我正在尝试将 libbeanstalkd 集成到嵌入式系统中,因此我必须进行一些小改动。到目前为止,我已经能够替换/修复一些特定于操作系统的代码,但后来我遇到了这个错误。我已经在 "expression must be a pointer to a complete object type" 之前修复了这些类型的错误,但是除了我确信宏和使用的对象的标头是该宏在抛出编译器错误之前的几行中使用。有人可以帮助我了解问题所在吗?

来源arrayqueue.h

#ifndef ARRAYQUEUE_H
#define ARRAYQUEUE_H 

#define AQ_DEFINE_STRUCT(struct_name, node_type) \
struct struct_name {                             \
    node_type *nodes;                            \
    size_t     size;                             \
    size_t     used;                             \
    off_t      front;                            \
    off_t      rear;                             \
}

#define AQ_NODES_FREE(q) ( (q)->size - (q)->used )
#define AQ_FULL(q)       ( (q)->used == (q)->size )
#define AQ_EMPTY(q)      ( (q)->used == 0 )

#define AQ_REAR_(q)      ( (q)->nodes + (q)->rear )
#define AQ_REAR(q)       ( AQ_EMPTY(q) ? NULL : AQ_REAR_(q) )

#define AQ_FRONT_(q)     ( (q)->nodes + (q)->front )
#define AQ_FRONT(q)      ( AQ_FULL(q) ? NULL : AQ_FRONT_(q) )

#define AQ_DEQ_FIN(q)    ( (q)->rear  = ( (q)->rear  + 1 ) % (q)->size, (q)->used-- )
#define AQ_ENQ_FIN(q)    ( (q)->front = ( (q)->front + 1 ) % (q)->size, (q)->used++ )

#endif /* ARRAYQUEUE_H */

来源ioqueue.h

#ifndef _IOQUEUE_H
#define _IOQUEUE_H

#include <stddef.h>
#include <vsocket.h>
#include "arrayqueue.h"
#include "verixmissing.h"

struct _ioq_node {
    struct iovec *vec;
    int    autofree;
};

AQ_DEFINE_STRUCT(_ioq, struct _ioq_node);

typedef struct _ioq ioq;
typedef struct _ioq_node ioq_node;

#define IOQ_NODES_READY(q)   ((q)->used ? ( (q)->front <= (q)->rear ? (q)->size - (q)->rear : (q)->used ) : 0)
#define IOQ_PEEK_POS(q, i)   ((AQ_REAR_(q)+i)->vec)
#define IOQ_REAR_(q)         IOQ_PEEK_POS(q,0)
#define IOQ_REAR(q)          (AQ_EMPTY(q) ? NULL : IOQ_REAR_(q))

void    ioq_enq_(ioq *q, void *data, ssize_t data_len, int autofree);
int     ioq_enq(ioq *q, void *data, ssize_t data_len, int autofree);
ssize_t ioq_dump(ioq *q, int fd);
ioq    *ioq_new(size_t size);
void    ioq_free(ioq *q);

#endif /* _IOQUEUE_H */

来源ioqueue.c

#include <stdlib.h>
#include <vsocket.h>
#include "arrayqueue.h"
#include "ioqueue.h"

...

ssize_t ioq_dump(ioq *q, int fd)
{
    size_t  bytes_expected = 0, nodes_ready = IOQ_NODES_READY(q), i;
    ssize_t bytes_written, nodes_written  = 0;

    for (i = 0; i < nodes_ready; ++i)
        bytes_expected += IOQ_PEEK_POS(q,i)->iov_len;

    if ( ( bytes_written = writev(fd, IOQ_REAR_(q), nodes_ready) ) < bytes_expected )
        switch (bytes_written) {
            case -1:
                return -1;
            default:
                while ( ( bytes_written -= IOQ_REAR_(q)->iov_len ) > 0 ) {
                    IOQ_DUMP_FIN(q, 1);
                    ++nodes_written;
                }
                if ( bytes_written < 0 ) {
                    IOQ_REAR_(q)->iov_base += IOQ_REAR_(q)->iov_len + bytes_written;
                    ^---------------------- ERROR!
                    IOQ_REAR_(q)->iov_len   = -bytes_written;
                }
                return nodes_written;
        }
    else {
        IOQ_DUMP_FIN(q, nodes_ready);
        return nodes_ready;
    }
}

错误:

"..\Source\libs\libbeanstalkd\ioqueue.c", line 125: Error:  #852: expression must be a pointer to a complete object type
                    IOQ_PEEK_POS(q, 0)->iov_base += IOQ_REAR_(q)->iov_len + bytes_written;
                    ^

如果有帮助,Eclipse 会将宏扩展为:((( (q)-&gt;nodes + (q)-&gt;rear )+0)-&gt;vec)

【问题讨论】:

  • AQ_REAR_()宏函数在哪里?添加问题
  • @GrijeshChauhan 在arrayqueue.h,添加源

标签: c macros beanstalkd


【解决方案1】:

struct iovec 可能未在您的代码中以某种方式定义。

struct _ioq_node {
    struct iovec *vec; // <--------
    int    autofree;
};

转储预处理器输出并检查定义。使用 gcc 命令行并将 -c 选项替换为 -E(并将 -o 设置为您希望输出到的任何位置)。

gcc -E -o <op> ...

【讨论】:

    【解决方案2】:

    所以我找到了问题...看来真正的问题是此方法试图增加套接字缓冲区的地址,但结构的 iov_base 定义为 void * 所以编译器不知道如何增加。 (从Error: expression must be a pointer to a complete object type (?)得到答案)

    iovec 声明:

    struct iovec
    {
        void *iov_base; // starting address of buffer
        int iov_len;  // size of buffer
    };
    

    解决方案是将其转换为已知类型,但不幸的是我不知道类型...

    【讨论】:

      猜你喜欢
      • 2012-06-14
      • 1970-01-01
      • 2012-01-02
      • 1970-01-01
      • 2023-03-11
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2014-08-19
      相关资源
      最近更新 更多