我正在尝试将libbeanstalkd集成到嵌入式系统中,所以我不得不做一些小改动。到目前为止,我已经能够替换/修复一些特定于操作系统的代码,但后来我遇到了这个错误。我已经修复了这些类型的错误,之后 “表达式必须是一个指向完整对象类型的指针” 但是真正让我感到沮丧的是我确信宏和使用过的对象的标头是在抛出编译器错误的前几行使用此宏。有人可以帮我理解问题是什么吗?
来源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)->nodes + (q)->rear )+0)->vec)
答案 0 :(得分:0)
您的代码可能无法以某种方式定义 struct iovec 。
struct _ioq_node {
struct iovec *vec; // <--------
int autofree;
};
转储预处理器输出并检查定义。使用gcc命令行并将-c选项替换为-E(并将-o设置为您想要输出的位置)。
gcc -E -o <op> ...
答案 1 :(得分:0)
所以我发现了问题......看来真正的问题是这个方法试图增加套接字缓冲区的地址,但是结构的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
};
解决方案是将其转换为已知类型但不幸的是我不知道类型......