`
yidongkaifa
  • 浏览: 4060342 次
文章分类
社区版块
存档分类
最新评论

android bionic缺失pthread_cancel的解决方法

 
阅读更多

在native code中使用多线程好处多多,但是Android的bionic并没有完全实现标准POSIX线程库的所有API,例如pthread_cancel()。但是google这样做肯定有原因,被cancel的thread不一定已经把自己拥有的资源释放掉,因此很可能带来内存泄露,锁没有释放等问题。这些问题在移动设备上更加突出。


首先介绍一个指标的方法,使用signal替代cancel调用:

当worker thread超时时,在主线程(或者是监视进程)中调用

  1. if((status=pthread_kill(pthread_id,SIGUSR1))!=0)
  2. {
  3. printf("Errorcancellingthread%d,error=%d(%s)",pthread_id,status,strerrorstatus));
  4. }

在worker thread中加入对SIGUSR1信号的处理

  1. structsigactionactions;
  2. memset(&actions,0,sizeof(actions));
  3. sigemptyset(&actions.sa_mask);
  4. actions.sa_flags=0;
  5. actions.sa_handler=thread_exit_handler;
  6. rc=sigaction(SIGUSR1,&actions,NULL);
  7. voidthread_exit_handler(intsig)
  8. {
  9. printf("thissignalis%d\n",sig);
  10. pthread_exit(0);
  11. }
参考自:http://stackoverflow.com/questions/4610086/pthread-cancel-alternatives-in-android-ndk


最根本的解决方法是重写worker thread,使用poll或者select等处理IO操作防止stuck的发生,下面是Android源码system/libsysutils/src/SocketListener.cpp的处理方法

1,创建worker thread前先创建通讯管道

  1. if(pipe(mCtrlPipe)){
  2. SLOGE("pipefailed(%s)",strerror(errno));
  3. return-1;
  4. }
  5. if(pthread_create(&mThread,NULL,SocketListener::threadStart,this)){
  6. SLOGE("pthread_create(%s)",strerror(errno));
  7. return-1;
  8. }


2,在worker thread的大循环中使用select同时监控管道和IO fd

  1. while(1){//一般工作进程都带一个大循环
  2. FD_SET(mCtrlPipe[0],&read_fds);
  3. if(mCtrlPipe[0]>max)
  4. max=mCtrlPipe[0];
  5. if((rc=select(max+1,&read_fds,NULL,NULL,NULL))<0){
  6. SLOGE("selectfailed(%s)",strerror(errno));
  7. sleep(1);
  8. continue;
  9. }elseif(!rc)
  10. continue;
  11. //有人喊停了
  12. if(FD_ISSET(mCtrlPipe[0],&read_fds))
  13. break;
  14. }


3,需要退出时通过管道通知worker thread

  1. if(write(mCtrlPipe[1],&c,1)!=1){
  2. SLOGE("Errorwritingtocontrolpipe(%s)",strerror(errno));
  3. return-1;
  4. }

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics