summaryrefslogtreecommitdiff
path: root/accel-pptpd/triton/md.c
diff options
context:
space:
mode:
authorKozlov Dmitry <dima@server>2010-08-25 19:50:32 +0400
committerKozlov Dmitry <dima@server>2010-08-25 19:50:32 +0400
commit14763d00a0777b7a27eb49eb5f91ac802c05ecb5 (patch)
treedf01692c930c8fe5d981e7931e41df19f234dc7e /accel-pptpd/triton/md.c
parent6f071c1e1ffeea801374416e38a9d14ee393ae13 (diff)
downloadaccel-ppp-xebd-14763d00a0777b7a27eb49eb5f91ac802c05ecb5.tar.gz
accel-ppp-xebd-14763d00a0777b7a27eb49eb5f91ac802c05ecb5.zip
rewriting triton library...
Diffstat (limited to 'accel-pptpd/triton/md.c')
-rw-r--r--accel-pptpd/triton/md.c394
1 files changed, 97 insertions, 297 deletions
diff --git a/accel-pptpd/triton/md.c b/accel-pptpd/triton/md.c
index a0d2fab..ac0fe00 100644
--- a/accel-pptpd/triton/md.c
+++ b/accel-pptpd/triton/md.c
@@ -6,337 +6,137 @@
#include "triton_p.h"
-#define USE_GET_TIME
+int max_events=128;
-static __thread struct list_head handlers;
-static __thread fd_set read_fds;
-static __thread fd_set write_fds;
-static __thread fd_set read_fds0;
-static __thread fd_set write_fds0;
-static __thread int md_term;
+static int epoll_fd;
+static struct epoll_event *epoll_events;
-asm(".hidden md_init");
-asm(".hidden md_run");
-asm(".hidden md_terminate");
+static pthread_t md_thr;
+static void* md_thread(void *arg)
-static void _triton_process_events(int wait);
-
-void md_init()
+int md_init()
{
- INIT_LIST_HEAD(&handlers);
-
- FD_ZERO(&read_fds);
- FD_ZERO(&write_fds);
-
signal(SIGPIPE,SIG_IGN);
- #ifdef USE_CORO
- triton_coroutine_create(0,md_run,0,1);
- #endif
-}
-void md_run()
-{
- md_term=0;
+ epoll_fd=epoll_create(0);
+ if (epoll_fd<0)
+ {
+ perror("epoll_create");
+ return -1;
+ }
- while(!md_term)
+ epoll_events=malloc(MAX_EVENTS * sizeof(struct epoll_event));
+ if (!epoll_events)
{
- _triton_process_events(1);
+ fprintf(stderr,"cann't allocate memory\n");
+ return -1;
}
-}
+ default_ctx=malloc(sizeof(*default_ctx));
+ if (!default_ctx)
+ {
+ fprintf(stderr,"cann't allocate memory\n");
+ return -1;
+ }
-#ifdef USE_CORO
-asm(".hidden cur_uc");
-ucontext_t cur_uc;
-#endif
+ triton_register_ctx(default_ctx);
-static void _triton_process_events(int wait)
+ return 0;
+}
+void md_run()
+{
+ pthread_create(&md_thr,md_thread,NULL);
+}
+
+static void* md_thread(void *arg)
{
int max_fd=0,t;
- struct md_handler_t *md_h;
+ struct triton_md_handler_t *h;
struct timeval tv1,tv2,twait0;
struct list_head *p1,*p2;
- int twait,n;
- int _break=0;
+ int timeout,i,n;
- gettimeofday(&tv1,NULL);
- _break=0;
-
- if (wait)
- {
- twait=timer_prepare(&tv1);
- #ifdef USE_CORO
- t=coroutine_get_timeout(&tv1);
- #else
- t=-1;
- #endif
- if (t>=0 && (twait==-1 || t<twait)) twait=t;
-
- list_for_each_entry(md_h,&handlers,entry)
- {
- if (md_h->in_handler) continue;
- if (md_h->handler->twait>=0 && (twait==-1 || md_h->handler->twait<twait)) twait=md_h->handler->twait;
- }
- }else
- {
- twait=0;
- }
-
- read_fds0=read_fds; write_fds0=write_fds;
-
- list_for_each_entry(md_h,&handlers,entry)
- {
- if (md_h->in_handler)
- {
- FD_CLR(md_h->fd,&read_fds0);
- FD_CLR(md_h->fd,&write_fds0);
- }else
- {
- if (md_h->fd>max_fd) max_fd=md_h->fd;
- }
- }
-
- twait0=(struct timeval){twait/1000,(twait%1000)*1000};
- n=select(max_fd+1,&read_fds0,&write_fds0,NULL,twait>=0?&twait0:NULL);
-
- gettimeofday(&tv2,NULL);
- twait=(tv2.tv_sec-tv1.tv_sec)*1000+(tv2.tv_usec-tv1.tv_usec)/1000;
-
- list_for_each_safe(p1,p2,&handlers)
- {
- md_h=list_entry(p1,struct md_handler_t,entry);
- //if (!md_h->del)
- {
- if (md_h->handler->twait>=0)
- {
- md_h->handler->twait-=twait;
- if (md_h->handler->twait<=0) md_h->timeout=1;
- }
- }
- }
-
- timer_check(&tv2);
- gettimeofday(&tv2,NULL);
- #ifdef USE_CORO
- coroutine_check_timeout(&tv2);
- #endif
-
- list_for_each_safe(p1,p2,&handlers)
- {
- md_h=list_entry(p1,struct md_handler_t,entry);
- if (md_h->in_handler) continue;
- if (!md_h->del)
- {
- if (md_h->timeout)
- {
- md_h->timeout=0;
- #ifdef USE_CORO
- md_h->in_handler=1;
- if (md_h->coro)
- {
- long int id=(long int)md_h->coro;
- md_h->coro=NULL;
- triton_coroutine_wakeup(id);
- }else
- #endif
- {
- md_h->handler->timeout(md_h->handler);
- }
- md_h->in_handler=0;
- if (_break) return;
- }
- }
- }
-
- if (n<0)
- {
- perror("triton: md(select)");
- //goto check_timeout;
- }
- if (n>0)
- {
- list_for_each_safe(p1,p2,&handlers)
- {
- md_h=list_entry(p1,struct md_handler_t,entry);
- if (md_h->in_handler) continue;
- if (md_h->del) continue;
- md_h->in_handler=1;
- if (FD_ISSET(md_h->fd,&read_fds0))
- {
- if (md_h->handler->read==md_h->handler->write)
- FD_CLR(md_h->fd,&write_fds0);
-
- #ifdef USE_CORO
- if (md_h->coro)
- {
- long int id=(long int)md_h->coro;
- md_h->coro=NULL;
- triton_coroutine_wakeup(id);
- }else
- #endif
- {
- md_h->handler->read(md_h->handler);
- }
- }
- if (!md_h->del && FD_ISSET(md_h->fd,&write_fds0) && md_h->handler->write)
- {
- #ifdef USE_CORO
- if (md_h->coro)
- {
- long int id=(long int)md_h->coro;
- md_h->coro=NULL;
- triton_coroutine_wakeup(id);
- }else
- #endif
- {
- md_h->handler->write(md_h->handler);
- }
- }
- md_h->in_handler=0;
- if (_break) return;
- }
- }
-//check_timeout:
-
- for(p1=handlers.next; p1!=&handlers;)
- {
- md_h=list_entry(p1,struct md_handler_t,entry);
- p1=p1->next;
- if (md_h->del)
- {
- list_del(&md_h->entry);
- free(md_h);
- }
- }
-
- if (!wait) _break=1;
-}
-
-void triton_process_events(void)
-{
- _triton_process_events(0);
+ n=epoll_wait(epoll_fd,epoll_events,MAX_EVENTS,-1);
+ if (n<0)
+ {
+ if (errno!=EINTR)
+ perror("epoll_wait");
+ continue;
+ }
+ if (n==0)
+ return;
+
+ for(i=0; i<n; i++)
+ {
+ h=(struct triton_md_handler_t*)epoll_events[i].data.ptr;
+ pthread_mutex_lock(&h->ctx->lock);
+ h->trig_epoll_events=epoll_events[i].events;
+ list_add_tail(&h->entry2,&h->ctx->pending_handlers);
+ h->pending=1;
+ triton_queue_ctx(h->ctx);
+ pthread_mutex_unlock(&h->ctx->lock);
+ }
}
void md_terminate()
{
- md_term=1;
+
}
void triton_md_register_handler(struct triton_md_handler_t *h)
{
- struct md_handler_t *md_h;
-
- list_for_each_entry(md_h,&handlers,entry)
- {
- if (md_h->handler==h)
- {
- if (!md_h->del)
- {
- printf("triton: bug: double triton_md_register_handler\n");
- abort();
- }
- md_h->del=0;
- md_h->in_handler=0;
- md_h->coro=0;
- md_h->fd=0;
- return;
- }
- }
-
- md_h=(struct md_handler_t *)malloc(sizeof(struct md_handler_t));
- memset(md_h,0,sizeof(*md_h));
- md_h->handler=h;
-
- list_add_tail(&md_h->entry,&handlers);
+ h->epoll_event.data.ptr=h;
+ if (!h->ctx)
+ h->ctx=default_ctx;
+ pthread_mutex_lock(&h->ctx->lock);
+ list_add_tail(&h->entry,&h->ctx->handlers);
+ pthread_mutex_unlock(&h->ctx->lock);
}
void triton_md_unregister_handler(struct triton_md_handler_t *h)
{
- struct md_handler_t *md_h;
-
- list_for_each_entry(md_h,&handlers,entry)
- {
- if (md_h->handler==h)
- {
- triton_md_disable_handler(h,0);
- /*list_del(&md_h->entry);
- free(md_h);
- return;*/
- md_h->del=1;
- return;
- }
- }
+ pthread_mutex_lock(&h->ctx->lock);
+ list_del(&h->entry);
+ if (h->pending)
+ list_del(&h->entry2);
+ pthread_lock_unlock(&h->ctx->lock);
}
-void triton_md_enable_handler(struct triton_md_handler_t *h, int mode)
+int triton_md_enable_handler(struct triton_md_handler_t *h, int mode)
{
- struct md_handler_t *md_h;
+ int r;
+ int events=h->epoll_event.events;
- list_for_each_entry(md_h,&handlers,entry)
- {
- if (md_h->handler==h)
- {
- md_h->fd=h->fd;
- break;
- }
- }
- if (mode)
- {
- if (mode&MD_MODE_READ)
- FD_SET(h->fd,&read_fds);
- if (mode&MD_MODE_WRITE)
- FD_SET(h->fd,&write_fds);
- }else
- {
- FD_SET(h->fd,&read_fds);
- FD_SET(h->fd,&write_fds);
- }
-}
-void triton_md_disable_handler(struct triton_md_handler_t *h,int mode)
-{
- if (mode)
- {
- if (mode&MD_MODE_READ)
- FD_CLR(h->fd,&read_fds);
- if (mode&MD_MODE_WRITE)
- FD_CLR(h->fd,&write_fds);
- }else
- {
- FD_CLR(h->fd,&read_fds);
- FD_CLR(h->fd,&write_fds);
- }
+ if (mode&MD_MODE_READ)
+ h->epoll_event.events|=EPOLLIN;
+ if (mode&MD_MODE_WRITE)
+ h->epoll_event.events|=EPOLLOUT;
+
+ h->epoll_event.events|=EPOLLET;
+
+ if (events)
+ r=epoll_ctl(epoll_fd,EPOLL_CTL_MOD,h->fd,&h->epoll_event);
+ else
+ r=epoll_ctl(epoll_fd,EPOLL_CTL_ADD,h->fd,&h->epoll_event);
+
+ return r;
}
-
-#ifdef USE_CORO
-int triton_md_wait(struct triton_md_handler_t *h)
+int triton_md_disable_handler(struct triton_md_handler_t *h,int mode)
{
- struct md_handler_t *md_h;
- int res=0;
-
- list_for_each_entry(md_h,&handlers,entry)
+ if (h->epoll_events.events)
+ return -1;
+
+ if (mode&MD_MODE_READ)
+ h->epoll_event.events&=~EPOLLIN;
+ if (mode&MD_MODE_WRITE)
+ h->epoll_event.events&=~EPOLLOUT;
+
+ if (h->epoll_event.events&(EPOLLIN|EPOLLOUT))
+ r=epoll_ctl(epoll_fd,EPOLL_CTL_MOD,h->fd,&h->epoll_event);
+ else
{
- if (md_h->handler==h) break;
+ h->epoll_event.events=0;
+ r=epoll_ctl(epoll_fd,EPOLL_CTL_DEL,h->fd,NULL);
}
- md_h->in_handler=0;
-
- md_h->coro=current_coro;
- triton_coroutine_schedule();
-
- if (FD_ISSET(md_h->fd,&read_fds0)) res|=MD_MODE_READ;
- if (FD_ISSET(md_h->fd,&write_fds0)) res|=MD_MODE_WRITE;
- return res;
-}
-int triton_md_wait2(int fd,int mode,int timeout)
-{
- int r;
- struct triton_md_handler_t h=
- {
- .fd=fd,
- .twait=timeout,
- };
- triton_md_register_handler(&h);
- triton_md_enable_handler(&h,mode);
- r=triton_md_wait(&h);
- triton_md_unregister_handler(&h);
- return r;
+ return r;
}
-#endif
+