NDK下IPC问题
由于AllJoyn的join session timeout问题一直无法解决,我们怀疑AllJoyn有些内部变量没有清理干净,因此考虑将AllJoyn相关功能放到一个单独的进程中,一旦join session timeout,就重启该进程。
这就涉及到IPC问题。
因为我们写的是通用模块,需要在DTV(arm平台linux)和Android(NDK)上运行,因此就必须考虑各种IPC手段的可用性了。
一开始我考虑直接使用socket进行IPC(肯定可用),但有人说消息队列、共享内存、管道之类的也可以用。于是俺就mmap+信号量实现了一个简单的IPC类:
/*
* File: ipc.h
* Author: raozf
*
* Created on 2013年9月17日, 上午11:04
*/ #ifndef IPC_H
#define IPC_H #include <semaphore.h>
#include <sys/mman.h>
#include <sys/stat.h>
#include <fcntl.h> //semaphore name
const std::string IPC_SEM = "alljoyn_sem";
//mmap length
const size_t IPC_MMAP_LEN = * * ; /*
* helper class for IPC between ****forked**** processes.
* implemented by semaphore and mmap
*/
class IPC
{
public:
IPC():_mutex(NULL), _memory(NULL)
{
//unnamed semaphore should use sem_init() //open named semaphore, shared between processes
_mutex = sem_open(IPC_SEM.c_str(), O_CREAT, O_RDWR, );
if(_mutex == SEM_FAILED || _mutex == NULL)
{
LOGV("[ContextSharing]IPC::IPC() sem_open() failed. semaphore name:%s, %s", IPC_SEM.c_str(), strerror(errno));
return;
} /*
* http://www.ibm.com/developerworks/cn/linux/l-ipc/part5/index1.html
*/
//create a anonymous mmap
_memory = (char*)mmap(NULL, IPC_MMAP_LEN, PROT_READ | PROT_WRITE, MAP_SHARED | MAP_ANONYMOUS, , );
if(_memory == MAP_FAILED || _memory == NULL)
{
LOGV("[ContextSharing]IPC::IPC() mmap() failed. %s", strerror(errno));
deinit();
return;
}
} ~IPC()
{
deinit();
} //send msg to other process
//header+message
// int char*
void send(const std::string& msg)
{
*((int*)_memory) = msg.length();
memcpy(_memory + sizeof(int), msg.c_str(), msg.length());
sem_post(_mutex);
} //receive from other process
std::string recv()
{
sem_wait(_mutex);//will block
return std::string(_memory + sizeof(int), (int)(_memory));
} private:
void deinit()
{
if(_mutex != NULL)
{
/* destory an unnamed semaphore, initialized by sem_init()
//sem_destroy(_mutex);
*/ // closes the named semaphore, but still existed in system kernel
sem_close(_mutex);
_mutex = NULL; //remove semaphore from system kernel(if it's reference is 0--which decremented by sem_close())
//removes the named semaphore referred to by name. The semaphore name is removed immediately.
//The semaphore is destroyed once all other processes that have the semaphore open close it.
sem_unlink(IPC_SEM.c_str());
} if(_memory != NULL)
{
munmap(_memory, IPC_MMAP_LEN);
_memory = NULL;
}
} private:
sem_t* _mutex;
char* _memory;
}; #endif /* IPC_H */
辅助类,封装一个线程专门等待信号量、接收数据:
/*
* File: ThreadedIPC.h
* Author: raozf
*
* Created on 2013年9月17日, 下午3:21
*/ #ifndef THREADEDIPC_H
#define THREADEDIPC_H #include "Common/ThreadManager.h"
#include "delegate.h"
#include "ipc.h" static const OID OID_ThreadedIPC = { 0x8f52a31f, 0x51d5, 0x462b, { 0xa9, 0x8d, 0x40, 0x70, 0xa3, 0xf6, 0xb5, 0x7f } };
class ThreadedIPC
:public CObjectRoot<CObjectMultiThreadModel>
,public IThreadClient
{
public:
PSFRESULT FinalConstruct()
{
PSFRESULT res = PSF_E_FAIL; CHK_PSFRESULT(_listen_thread.Initialize());
_listen_thread.AddTask(this, NULL); CLEANUP:
return res;
} void FinalRelease()
{
_listen_thread.Terminate();
} void Send(const std::string& msg)
{
_sharer.send(msg);
} private:
void OnExecute(void* pArg)
{
while(true)
{
_delegate_recv(_sharer.recv());
}
} void OnTerminate(void* pArg)
{
} public:
delegate::Delegate<void(std::string)> _delegate_recv; private:
CTaskWorker _listen_thread;
IPC _sharer; BEGIN_OBJECT_INTERFACE_MAP()
OBJECT_INTERFACE_ENTRY(OID_ThreadedIPC, this);
END_OBJECT_INTERFACE_MAP()
}; #endif /* THREADEDIPC_H */
这里面用到了我们自己写的delegate和thread类(先忽略之)。
但在android上运行时却报错:
IPC::IPC() sem_open() failed. semaphore name:alljoyn_sem, Function not implemented
看来信号量在NDK下是用不了的。
而且,不仅仅信号量,共享内存、消息队列在NDK下都不能用。
参见讨论:https://groups.google.com/forum/#!topic/android-ndk/FzJIsJIxCX4
http://stackoverflow.com/questions/18603267/cross-process-locking-with-android-ndk
(但管道可用?http://stackoverflow.com/questions/8471552/porting-c-code-which-uses-fork-to-android)
Android源代码中的文档说明:http://www.netmite.com/android/mydroid/1.6/bionic/libc/docs/SYSV-IPC.TXT
(该文件在Android4。3中似乎已经移出该文档)
Android does not support System V IPCs, i.e. the facilities provided by the
following standard Posix headers: <sys/sem.h> /* SysV semaphores */
<sys/shm.h> /* SysV shared memory segments */
<sys/msg.h> /* SysV message queues */
<sys/ipc.h> /* General IPC definitions */ The reason for this is due to the fact that, by design, they lead to global
kernel resource leakage.
原因就是防止内核资源泄露。
更重要的问题在于:fork()也尽量不要用。
道理很简单:我们不应该控制android的底层,这些api会造成系统的不稳定。
https://groups.google.com/forum/#!msg/android-platform/80jr-_A-9bU/nkzslcgVrfYJ
“Bear in mind that the Dalvik VM doesn't like fork() much, and goes into
conniptions if you try to do *any* work between the fork() and the
exec(). ”
https://groups.google.com/forum/#!topic/android-ndk/FzJIsJIxCX4
悲催!
----------------------------
ps:学到了一个命令“ipcs”,查看系统中共享内存、信号量状态:
$ ipcs
------ Shared Memory Segments --------
key shmid owner perms bytes nattch status
------ Semaphore Arrays --------
key semid owner perms nsems
------ Message Queues --------
key msqid owner perms used-bytes messages
NDK下IPC问题的更多相关文章
- C++成员变量内存对齐问题,ndk下非对齐的内存访问导致BUS_ADRALN
同样的代码,在vs下运行正常,在android ndk下却崩溃: signal 7(SIGBUS),code 1 (BUS_ADRALN),fault addr 0xe6b82793 Func(sho ...
- Android NDK 下的宽字符编码转换及icu库的使用(转)
原贴http://topic.csdn.net/u/20101022/16/1b2e0cec-b9d2-42ea-8d9c-4f1bb8320a54.html?r=70149216 ,看过并动手实现, ...
- 20155202 张旭 课下作业: Linux下IPC机制
20155202张旭 Linux下IPC机制 IPC机制定义 在linux下的多个进程间的通信机制叫做IPC(Inter-Process Communication),它是多个进程之间相互沟通的一种方 ...
- 20155239吕宇轩 Linux下IPC机制
20155239吕宇轩 Linux下IPC机制 - 共享内存 原理:把所有需要使用的共享数据都存放在共享内存 区域中,任何想要访问这些共享数据的进程都必须在自己的进程地址空间中新增加一块内存区域,用来 ...
- 2017-2018-1 20155307 《信息安全系统设计基础》第十周课上未完成补充以及课下IPC作业
课上内容2:stat命令的实现-mysate 学习使用stat(1),并用C语言实现 提交学习stat(1)的截图 man -k ,grep -r的使用 伪代码 产品代码 mystate.c,提交码云 ...
- Linux下IPC机制
Linux下IPC机制 实践要求 研究Linux下IPC机制:原理,优缺点,每种机制至少给一个示例,提交研究博客的链接 共享内存 管道 FIFO 信号 消息队列 IPC 进程间通信(IPC,Inter ...
- NDK下 将Platinum SDK 编译成so库 (android - upnp)
Platinum UPnP SDK 是一个跨平台的C++库,利用该库,可以很容易就构建出DLNA/UPnP控制点(DLNA/UPnP Control Point)和DLNA/UPnP设备(DLNA/U ...
- NDK下编译JNI
NDK环境下编译JNI 下载demo.tar.gz然后解压 弄个套路 1.编辑build.sh设置好NDK目录 2.把cpp文件放到code下面 运行sh build.sh即可
- 【记录一个问题】没用任何用处的解决了libtask的context.c在32位NDK下的编译问题
32位下用ndk编译libtask出现这样的错误: [armeabi-v7a] Compile thumb : task <= context.c /Users/ahfu/code/androi ...
随机推荐
- hdu 5576 dp
题目大意:给你一个长度为 n 的 字符串表示一个乘法,一次操作随机选两个字符进行交换,进行m次操作,让你求出所有可能操作 的答案和. (1 <= n, m <= 50) 思路:巨难.. ...
- 【剑指offer】面试题 17. 打印从 1 到最大的 n 位数
面试题 17. 打印从 1 到最大的 n 位数 题目描述 题目:输入数字 n,按顺序打印出从 1 最大的 n 位十进制数.比如输入 3,则打印出 1.2.3 一直到最大的 3 位数即 999. 解答过 ...
- Laravel 下配置 Redis 让缓存、Session 各自使用不同的 Redis 数据库
为什么要这样做? 默认情况下,Redis 服务会提供 16 个数据库,Laravel 使用数据库 0 (请见 Redis 文档)作为缓存和 Session 的存储. 在使用的过程中觉得这个默认的设置挺 ...
- thinkphp Upload上传文件在客户端生成的临时文件$_FILES['file']['tmp_name']
1.关于thinkphp 的Upload的$_FILES['file']['tmp_name'] 在使用thinkphp上传图片的时候,在上传的$_FILES数组中,有一个$_FILES['file' ...
- CodeForces 779E Bitwise Formula
位运算,枚举. 按按分开计算,枚举$?$是$0$还是$1$,分别计算出$sum$,然后就可以知道该位需要填$1$还是$0$了. #include<map> #include<set& ...
- Struts2中使用Session
嗯,没错,过了这么长时间我还在做我的注册和登录. 登陆的部分自然是从数据库中查找响应的用户信息,然后把一个user放到session里, Action中代码如下: public String Logi ...
- 洛谷——P1190 接水问题
P1190 接水问题 题目描述 学校里有一个水房,水房里一共装有 m 个龙头可供同学们打开水,每个龙头每秒钟的供水量相等,均为 1. 现在有 n 名同学准备接水,他们的初始接水顺序已经确定.将这些同学 ...
- Flask实战第54天:cms删除轮播图功能完成
后台逻辑 编辑cms.views.py @bp.route('/dbanner/',methods=['POST']) @login_required def dbanner(): banner_id ...
- 反编译apk 修改 合成
韩梦飞沙 韩亚飞 313134555@qq.com yue31313 han_meng_fei_sha 反编译apk帮助文档 准备工具 dex2jar(dex转换jar工具),下载地址: ht ...
- Codeforces 804D Expected diameter of a tree(树形DP+期望)
[题目链接] http://codeforces.com/contest/804/problem/D [题目大意] 给你一个森林,每次询问给出u,v, 从u所在连通块中随机选出一个点与v所在连通块中随 ...