1 Star 0 Fork 15

nbsnail/beanstalkd-win

forked from 蔡东赟/beanstalkd-win 
加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
windows.c 6.57 KB
一键复制 编辑 原始数据 按行查看 历史
caidongyun 提交于 2015-03-20 14:09 . beanstalkd windows support
#include <stdint.h>
#include <sys/types.h>
#include <unistd.h>
#include <errno.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/select.h>
#include <sys/socket.h>
//#include <sys/event.h>
#include <sys/time.h>
#include "dat.h"
typedef struct fd_list {
u_int fd_count; /* how many are SET? */
Socket* fd_array[FD_SETSIZE]; /* an array of SOCKETs */
} fd_list;
#define FDLIST_CLR(fd, set) do { \
u_int __i; \
for (__i = 0; __i < ((fd_list *)(set))->fd_count ; __i++) { \
if (((fd_list *)(set))->fd_array[__i] == (fd)) { \
while (__i < ((fd_list *)(set))->fd_count-1) { \
((fd_list *)(set))->fd_array[__i] = \
((fd_list *)(set))->fd_array[__i+1]; \
__i++; \
} \
((fd_list *)(set))->fd_count--; \
break; \
} \
} \
} while(0)
#define FDLIST_SET(fd, set) do { \
u_int __i; \
for (__i = 0; __i < ((fd_list *)(set))->fd_count; __i++) { \
if (((fd_list *)(set))->fd_array[__i] == (fd)) { \
break; \
} \
} \
if (__i == ((fd_list *)(set))->fd_count) { \
if (((fd_list *)(set))->fd_count < FD_SETSIZE) { \
((fd_list *)(set))->fd_array[__i] = (fd); \
((fd_list *)(set))->fd_count++; \
} \
} \
} while(0)
static fd_set master_read_fd_set;
static fd_set master_write_fd_set;
static fd_set work_read_fd_set;
static fd_set work_write_fd_set;
static fd_list master_read_fd_list;
static fd_list master_write_fd_list;
static fd_list work_read_fd_list;
static fd_list work_write_fd_list;
static int max_fd;
static char buf0[512]; /* buffer of zeros */
static int
select_repair_fd_sets(Socket **ret)
{
int n, i;
int len;
Socket* s;
for (i = 0; i < master_read_fd_list.fd_count; i++) {
s = master_read_fd_list.fd_array[i];
len = sizeof(int);
if (getsockopt(s->fd, SOL_SOCKET, SO_TYPE, &n, &len) == -1) {
//printf("invalid descriptor %d in read fd_set", s->fd);
FD_CLR(s->fd, &master_read_fd_set);
FDLIST_CLR(s, &master_read_fd_list);
*ret = s;
return 'h';
}
}
for (i = 0; i < master_write_fd_list.fd_count; i++) {
s = master_write_fd_list.fd_array[i];
len = sizeof(int);
if (getsockopt(s->fd, SOL_SOCKET, SO_TYPE, &n, &len) == -1) {
// printf("invalid descriptor %d in write fd_set", s->fd);
FD_CLR(s->fd, &master_write_fd_set);
FDLIST_CLR(s, &master_write_fd_list);
*ret = s;
return 'h';
}
}
max_fd = -1;
return 0;
}
/* Allocate disk space.
* Expects fd's offset to be 0; may also reset fd's offset to 0.
* Returns 0 on success, and a positive errno otherwise. */
int
rawfalloc(int fd, int len)
{
// printf("rawfalloc len\n");
int i, w;
for (i = 0; i < len; i += w) {
w = write(fd, buf0, sizeof buf0);
if (w == -1) return errno;
}
lseek(fd, 0, 0); /* do not care if this fails */
return 0;
}
int
sockinit(void)
{
//printf("sockinit\n");
FD_ZERO(&master_read_fd_set);
FD_ZERO(&master_write_fd_set);
max_fd = -1;
return 0;
}
int
sockwant(Socket *s, int rw)
{
// printf("sockwant, fd:%d, want:%c\n", s->fd, (char)rw);
if (s->added) {
switch (s->added) {
case 'r':
//printf("close read, fd:%d\n", s->fd);
FD_CLR(s->fd, &master_read_fd_set);
FDLIST_CLR(s, &master_read_fd_list);
break;
case 'w':
//printf("close write, fd:%d\n", s->fd);
FD_CLR(s->fd, &master_write_fd_set);
FDLIST_CLR(s, &master_write_fd_list);
break;
default:
break;
}
}
if (rw) {
switch (rw) {
case 'r':
FD_SET(s->fd, &master_read_fd_set);
FDLIST_SET(s, &master_read_fd_list);
break;
case 'w':
FD_SET(s->fd, &master_write_fd_set);
FDLIST_SET(s, &master_write_fd_list);
break;
default:
break;
}
s->added = rw;
if (max_fd < s->fd) {
max_fd = s->fd;
}
}
return 0;
}
int
socknext(Socket **s, int64 timeout)
{
int r, i;
struct timeval tv, *tp;
struct fd_list fd_notified = {0, {0}};
// finish the list created before.
if (work_read_fd_list.fd_count != 0) {
*s = work_read_fd_list.fd_array[work_read_fd_list.fd_count-1];
FDLIST_CLR(*s, &work_read_fd_list);
// printf(" read socket before %d\n", (*s)->fd);
return 'r';
}
if (work_write_fd_list.fd_count != 0){
*s = work_write_fd_list.fd_array[work_write_fd_list.fd_count-1];
FDLIST_CLR(*s, &work_write_fd_list);
//printf(" write socket before %d\n", (*s)->fd);
return 'w';
}
work_read_fd_set = master_read_fd_set;
work_write_fd_set = master_write_fd_set;
work_read_fd_list = master_read_fd_list;
work_write_fd_list = master_write_fd_list;
tv.tv_sec = 0;//(long) (msec / 1000);
tv.tv_usec = 500000;//(long) (msec);
tp = &tv;
//printf("max_fd:%d, socknext () second:%ld. %ld\n", max_fd, tv.tv_sec, tv.tv_usec);
r = select(max_fd + 1, &work_read_fd_set, &work_write_fd_set, NULL, tp);
if (r == -1) {
twarnx("select erro");
return select_repair_fd_sets(s);
}
for (i = 0; i < work_read_fd_list.fd_count; i++) {
int fd = work_read_fd_list.fd_array[i]->fd;
if (FD_ISSET(fd, &work_read_fd_set) == 0) {
continue;
}
FDLIST_SET(work_read_fd_list.fd_array[i], &fd_notified);
}
work_read_fd_list = fd_notified;
fd_notified.fd_count = 0;
for (i = 0; i < work_write_fd_list.fd_count; i++) {
int fd = work_write_fd_list.fd_array[i]->fd;
if (FD_ISSET(fd, &work_write_fd_set) == 0) {
continue;
}
FDLIST_SET(work_write_fd_list.fd_array[i], &fd_notified);
}
work_write_fd_list = fd_notified;
if (work_read_fd_list.fd_count != 0) {
*s = work_read_fd_list.fd_array[work_read_fd_list.fd_count-1];
FDLIST_CLR(*s, &work_read_fd_list);
// printf(" read socket %d\n", (*s)->fd);
return 'r';
}
if (work_write_fd_list.fd_count != 0){
*s = work_write_fd_list.fd_array[work_write_fd_list.fd_count-1];
FDLIST_CLR(*s, &work_write_fd_list);
// printf(" write socket %d\n", (*s)->fd);
return 'w';
}
return 0;
}
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/nbsnail/beanstalkd-win.git
git@gitee.com:nbsnail/beanstalkd-win.git
nbsnail
beanstalkd-win
beanstalkd-win
master

搜索帮助