takephoto.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441
  1. #include "takephoto.h"
  2. #include "stb_image_write.h"
  3. // 模块名称
  4. static const char MODULE_NAME[] = "HrTakePhoto";
  5. // 相机类型
  6. static unsigned int device_type = interfaceTypeGige | interfaceTypeUsb3;
  7. // 制造厂商
  8. static const char * const manufacturer = "Huaray Technology";
  9. // 触发模式
  10. static const char * const setTriggerMode = "Off"; /// 默认: "Off", 可选: "On"
  11. // 曝光取图
  12. #define AE_WAIT_MAX_MS 180000 // 最大曝光稳定等待的时间, 默认3分钟, 单位: ms
  13. #define AE_EXP_TIME_EPS_US 50 // 曝光时间稳定判定的误差, 防抖&区间, 单位: us
  14. #define AE_EXP_STABLE_FRAMES 4 // 相机处于自动曝光模式时, 累计曝光值稳定的帧数
  15. // 保存照片
  16. static int SavePhoto(HANDLE hCam, IMV_Frame *pFrame, DHImgType imgType, const char *imgPath);
  17. // 拍照环境
  18. typedef struct
  19. {
  20. HANDLE hCam; // 打开相机的句柄
  21. uint32_t camType; // 相机类型:U、网
  22. char manuName[MAX_LINE_CHARS]; //制造厂商名称
  23. DHImgType saveImgType; // 保存图像的类型
  24. const char *saveImgPath; // 保存图像的路径
  25. bool isExposureAuto; // 自动曝光:是/否
  26. struct timespec expTime0; // 曝光开始的时间
  27. float lastExpTime; // 上次的曝光时长(us)
  28. int expStableCnt; // 连续曝光稳定帧计数
  29. HANDLE hESig; // 任务结束的通知
  30. int rCode; // 任务结束返回值
  31. void *pUser; // 可选的用户数据
  32. } PthotoProcCtx;
  33. // 非标准的"Huaray Technology"制造商名称, 但也可视为华睿的
  34. static const char * const huaray_manu_aliases[] = { "Machine Vision", NULL };
  35. static bool is_huaray_manu_alias(const char *name)
  36. {
  37. for(int i = 0; huaray_manu_aliases[i] != NULL; i++)
  38. {
  39. if(xstrcasecmp(name, huaray_manu_aliases[i]) == 0) return true;
  40. }
  41. return false;
  42. }
  43. static void OnFrameReceived(IMV_Frame *pFrame, void *pUser) // 数据帧的回调, 完成一次拍照任务
  44. {
  45. PthotoProcCtx *ctx = (PthotoProcCtx *)pUser; int ret = IMV_OK;
  46. if(NULL == pFrame) return;
  47. // 1, 打印该帧信息
  48. sw_log_debug("[%s] +++GetOneFrame+++, Width[%u], Height[%u], FrameNum[%llu], FrameLen[%u], PixelType[0x%08X]", MODULE_NAME,
  49. pFrame->frameInfo.width, pFrame->frameInfo.height, pFrame->frameInfo.blockId, pFrame->frameInfo.size,
  50. (unsigned int)pFrame->frameInfo.pixelFormat);
  51. // 2, 自动曝光模式
  52. if(ctx->isExposureAuto)
  53. {
  54. double curExpTime = 0.0;
  55. ret = IMV_GetDoubleFeatureValue(ctx->hCam, "ExposureTime", &curExpTime);
  56. if(IMV_OK != ret) goto end_p;
  57. struct timespec now; clock_gettime(CLOCK_MONOTONIC, &now);
  58. long elapsed = (now.tv_sec - ctx->expTime0.tv_sec)*1000 + (now.tv_nsec-ctx->expTime0.tv_nsec)/(1000*1000); // 计算时间差, 单位: ms
  59. sw_log_debug("[%s] +++GetOneFrame+++, 等待曝光完成, ExposureTime = %.2fus, ElapsedTime = %ldms", MODULE_NAME, curExpTime, elapsed);
  60. if(fabsf(curExpTime - ctx->lastExpTime) <= AE_EXP_TIME_EPS_US) ctx->expStableCnt++;
  61. else ctx->expStableCnt = 0;
  62. if(ctx->expStableCnt < AE_EXP_STABLE_FRAMES && elapsed < AE_WAIT_MAX_MS)
  63. { // 曝光未稳定, 也未超时, 则继续等下一帧
  64. ctx->lastExpTime = curExpTime; if(0 == ctx->expStableCnt) ctx->expStableCnt = 1;
  65. return;
  66. }
  67. }
  68. // 3, 导出图像文件
  69. if(ctx->pUser) { sw_thrd_destroy(ctx->pUser, WAITTHRD_SAFEEXIT_TIMEOUT); ctx->pUser = NULL; }
  70. ret = SavePhoto(ctx->hCam, pFrame, ctx->saveImgType, ctx->saveImgPath);
  71. // 4, 设置拍照完成
  72. end_p:
  73. ctx->rCode = ret; sw_signal_give(ctx->hESig);
  74. }
  75. static int FrameSoftTrigger(unsigned long wParam, unsigned long lParam) // 线程回调函数, 触发相机拍照
  76. {
  77. PthotoProcCtx *ctx = (PthotoProcCtx *)wParam;
  78. void *hThrd = ctx->pUser; int slot_ms = 100, tick_ms, ret;
  79. while(sw_thrd_isAlive(hThrd))
  80. {
  81. sw_log_debug("[%s] 触发一次拍照", MODULE_NAME);
  82. ret = IMV_ExecuteCommandFeature(ctx->hCam, "TriggerSoftware");
  83. if(IMV_OK != ret) { sw_log_error("[%s] 触发拍照失败, errCode=%d!!", MODULE_NAME, ret); break; }
  84. else tick_ms = 0;
  85. while(sw_thrd_isAlive(hThrd) && tick_ms < 1000) { sw_thrd_delay(slot_ms); tick_ms += slot_ms; }
  86. }
  87. return -1;
  88. }
  89. // 单次执行相机拍照, 并保存到文件, 成功返回: 0值, 失败返回:非0值
  90. // "imgType" - 获取图像类型
  91. // "saveImgPath" - 保存的文件名
  92. // "timeout" - 等待超时时间, 单位:秒: < 0 表示无超时;
  93. // 无论超时怎么设置, 首次尝试拍照一定会执行
  94. // , 但时间不确定; 拍照成功、超时或发生错误
  95. // 时会自动结束任务(相机持续无数据的时间).
  96. // "pImgMark" - 输出本次拍照图像的水印信息, 可以设置NULL
  97. int DH_TakePhoto(DHImgType imgType, const char *saveImgPath, int timeout, DHImgMark *pImgMark)
  98. {
  99. int fd; char runDir[MAX_PATH_CHARS] = { 0 }, lockFile[MAX_PATH_CHARS+32] = { 0 };
  100. int ret, index = 0; DHImgMark imgMark = { 0 }; HANDLE hCam = NULL; PthotoProcCtx ctx = { 0 };
  101. IMV_DeviceList devList = { 0 }; IMV_DeviceInfo *pDevInfo = NULL; IMV_String exposureMode = { 0 };
  102. // 1, 占用锁定, 避免同时间拍照
  103. xGetSelfRunningInfo(runDir, NULL);
  104. if(runDir[strlen(runDir)-1] == '/') sprintf(lockFile, "%s%s", runDir, "status/");
  105. else sprintf(lockFile, "%s/%s", runDir, "status/");
  106. if(!sw_dir_exists(lockFile)) sw_dir_create(lockFile);
  107. strcat(lockFile, "dh_takephoto.lock");
  108. fd = open(lockFile, O_CREAT | O_RDWR | __O_CLOEXEC, 0666);
  109. if(-1 == fd) { return -1; }
  110. if(-1 == flock(fd, LOCK_EX | LOCK_NB)) { close(fd); return -2; }
  111. // 2, 查找相机, 获取其设备信息
  112. ret = IMV_EnumDevices(&devList, device_type);
  113. if(IMV_OK != ret)
  114. {
  115. sw_log_error("[%s] 枚举相机,执行错误, errCode=%d!!", MODULE_NAME, ret);
  116. goto end_p;
  117. }
  118. if(devList.nDevNum < 1)
  119. {
  120. ret = -3;
  121. sw_log_error("[%s] 没有相机,数量=%u!!", MODULE_NAME, devList.nDevNum);
  122. goto end_p;
  123. }
  124. findp:
  125. pDevInfo = &devList.pDevInfo[index++];
  126. if(!pDevInfo)
  127. {
  128. ret = -4;
  129. sw_log_error("[%s] unexpected internal error, failed to obtain camera information!!", MODULE_NAME);
  130. goto end_p;
  131. }
  132. if(pDevInfo->nCameraType == typeU3vCamera) ctx.camType = interfaceTypeUsb3;
  133. else if(pDevInfo->nCameraType == typeGigeCamera) ctx.camType = interfaceTypeGige;
  134. else { // SDK出错, 我只枚举了U口和G口相机, 不可能出现其他类型
  135. ret = -5;
  136. sw_log_error("[%s] unexpected internal error, unknown camera type(%u)!!", MODULE_NAME, pDevInfo->nCameraType);
  137. goto end_p;
  138. }
  139. strcpy(ctx.manuName, pDevInfo->vendorName);
  140. strcpy(imgMark.camModelName, pDevInfo->modelName);
  141. strcpy(imgMark.camSerialNum, pDevInfo->serialNumber);
  142. if(xstrcasecmp(ctx.manuName, manufacturer) != 0 && !is_huaray_manu_alias(ctx.manuName))
  143. {
  144. if(index < devList.nDevNum) goto findp;
  145. ret = -6;
  146. sw_log_error("[%s] 没有找到 \"%s\" 的相机!!", MODULE_NAME, manufacturer);
  147. goto end_p;
  148. }
  149. // 3, 打开相机, 设置其触发模式
  150. unsigned int cameraIndex = (index - 1);
  151. ret = IMV_CreateHandle(&hCam, modeByIndex, (void*)&cameraIndex);
  152. if(IMV_OK == ret) ret = IMV_Open(hCam);
  153. if(IMV_OK != ret)
  154. {
  155. sw_log_error("[%s] 打开相机时发生错误, errCode=%d!!", MODULE_NAME, ret);
  156. goto end_p;
  157. }
  158. if(IMV_OK == ret && xstrcasecmp(setTriggerMode, "On") == 0) ret = IMV_SetEnumFeatureSymbol(hCam, "TriggerSelector", "FrameStart"); // 设置触发条件
  159. if(IMV_OK == ret && xstrcasecmp(setTriggerMode, "On") == 0) ret = IMV_SetEnumFeatureSymbol(hCam, "TriggerSource", "Software"); // 设置软件触发
  160. if(IMV_OK == ret) ret = IMV_SetEnumFeatureSymbol(hCam, "TriggerMode", setTriggerMode); // 设置触发模式
  161. if(IMV_OK != ret)
  162. {
  163. sw_log_error("[%s] 设置相机时发生错误, errCode=%d!!", MODULE_NAME, ret);
  164. goto end_p;
  165. }
  166. // 4, 开始拍照, 等待完成后输出
  167. ret = IMV_GetEnumFeatureSymbol(hCam, "ExposureAuto", &exposureMode);
  168. if(IMV_OK != ret)
  169. {
  170. sw_log_error("[%s] 获取曝光模式时出错, errCode=%d!!", MODULE_NAME, ret);
  171. goto end_p;
  172. }
  173. if(xstrcasecmp(exposureMode.str, "Off") != 0) { ctx.isExposureAuto = true; clock_gettime(CLOCK_MONOTONIC, &ctx.expTime0); }
  174. else ctx.isExposureAuto = false;
  175. ctx.hCam = hCam;
  176. ctx.saveImgType = imgType;
  177. ctx.saveImgPath = saveImgPath;
  178. ctx.hESig = sw_signal_create();
  179. if(!ctx.hESig)
  180. {
  181. ret = -7;
  182. sw_log_error("[%s] SIG信号量创建失败!!", MODULE_NAME);
  183. goto end_p;
  184. }
  185. ctx.rCode = IMV_OK;
  186. ret = IMV_AttachGrabbing(hCam, OnFrameReceived, (void *)&ctx);
  187. if(IMV_OK != ret)
  188. {
  189. sw_log_error("[%s] 注册回调时发生错误, errCode=%d!!", MODULE_NAME, ret);
  190. goto end_p;
  191. }
  192. ret = IMV_StartGrabbing(hCam);
  193. if(IMV_OK != ret)
  194. {
  195. sw_log_error("[%s] 相机取流时发生错误, errCode=%d!!", MODULE_NAME, ret);
  196. goto end_p;
  197. }
  198. HANDLE hThrd = NULL;
  199. if(0 != xstrcasecmp(setTriggerMode, "On")) goto waitp;
  200. hThrd = sw_thrd_create("PhotoProc", THREAD_DEFAULT_PRIORITY, THREAD_DEFAULT_STACK_SIZE, FrameSoftTrigger, (unsigned long)&ctx, 0);
  201. if(!hThrd)
  202. {
  203. ret = -8;
  204. sw_signal_destroy(ctx.hESig);
  205. sw_log_error("[%s] 触发线程-创建失败!!", MODULE_NAME);
  206. goto end_p;
  207. }
  208. ctx.pUser = (void *)hThrd; sw_thrd_resume(hThrd);
  209. waitp:
  210. ret = sw_signal_wait(ctx.hESig, timeout*1000); // 阻塞等待拍照任务结束或超时
  211. if(0 == ret) ret = ctx.rCode;
  212. else
  213. {
  214. ret = -9;
  215. sw_log_error("[%s] 拍照过程-等待超时!!", MODULE_NAME);
  216. }
  217. if(ctx.pUser) sw_thrd_destroy(ctx.pUser, WAITTHRD_SAFEEXIT_TIMEOUT);
  218. sw_signal_destroy(ctx.hESig);
  219. // 5, 成功拍照, 输出相机的信息
  220. if(IMV_OK == ret)
  221. {
  222. double curExpTime = 0.0;
  223. IMV_GetDoubleFeatureValue(hCam, "ExposureTime", &curExpTime);
  224. imgMark.imgExposureTime = curExpTime;
  225. if(pImgMark) memcpy(pImgMark, &imgMark, sizeof(DHImgMark));
  226. }
  227. // 6, 任务结束, 释放相关的资源
  228. end_p:
  229. if(hCam)
  230. {
  231. IMV_StopGrabbing(hCam);
  232. IMV_Close(hCam);
  233. IMV_DestroyHandle(hCam);
  234. }
  235. flock(fd, LOCK_UN);
  236. close(fd);
  237. sw_file_delete(lockFile);
  238. return ret;
  239. }
  240. // 保存给定的数据帧到BMP图像文件, 成功返回: 0值, 失败返回: 非0值
  241. static int SaveFrameToBmp(IMV_HANDLE hCam, IMV_Frame *pFrame, const char *path)
  242. {
  243. IMV_PixelConvertParam stPixelConvertParam; int ret = IMV_OK;
  244. unsigned char *pDstBuf = NULL; unsigned int nDstBufSize = 0;
  245. nDstBufSize = sizeof(unsigned char) * pFrame->frameInfo.width * pFrame->frameInfo.height * 3;
  246. pDstBuf = (unsigned char*)malloc(nDstBufSize);
  247. if(!pDstBuf) return -18;
  248. memset(&stPixelConvertParam, 0, sizeof(stPixelConvertParam));
  249. stPixelConvertParam.nWidth = pFrame->frameInfo.width;
  250. stPixelConvertParam.nHeight = pFrame->frameInfo.height;
  251. stPixelConvertParam.ePixelFormat = pFrame->frameInfo.pixelFormat;
  252. stPixelConvertParam.pSrcData = pFrame->pData;
  253. stPixelConvertParam.nSrcDataLen = pFrame->frameInfo.size;
  254. stPixelConvertParam.nPaddingX = pFrame->frameInfo.paddingX;
  255. stPixelConvertParam.nPaddingY = pFrame->frameInfo.paddingY;
  256. stPixelConvertParam.eBayerDemosaic = demosaicNearestNeighbor;
  257. stPixelConvertParam.eDstPixelFormat = gvspPixelRGB8;
  258. stPixelConvertParam.pDstBuf = pDstBuf;
  259. stPixelConvertParam.nDstBufSize = nDstBufSize;
  260. ret = IMV_PixelConvert(hCam, &stPixelConvertParam);
  261. if(IMV_OK != ret) goto ret_p;
  262. ret = stbi_write_bmp(path, pFrame->frameInfo.width, pFrame->frameInfo.height, 3, pDstBuf);
  263. if(1 == ret) ret = IMV_OK;
  264. else ret = -19;
  265. ret_p:
  266. if(pDstBuf) free(pDstBuf);
  267. return ret;
  268. }
  269. // 保存给定的数据帧到JPG图像文件, 成功返回: 0值, 失败返回: 非0值
  270. static int SaveFrameToJpg(IMV_HANDLE hCam, IMV_Frame *pFrame, const char *path)
  271. {
  272. IMV_PixelConvertParam stPixelConvertParam; int ret = IMV_OK;
  273. unsigned char *pDstBuf = NULL; unsigned int nDstBufSize = 0;
  274. nDstBufSize = sizeof(unsigned char) * pFrame->frameInfo.width * pFrame->frameInfo.height * 3;
  275. pDstBuf = (unsigned char*)malloc(nDstBufSize);
  276. if(!pDstBuf) return -18;
  277. memset(&stPixelConvertParam, 0, sizeof(stPixelConvertParam));
  278. stPixelConvertParam.nWidth = pFrame->frameInfo.width;
  279. stPixelConvertParam.nHeight = pFrame->frameInfo.height;
  280. stPixelConvertParam.ePixelFormat = pFrame->frameInfo.pixelFormat;
  281. stPixelConvertParam.pSrcData = pFrame->pData;
  282. stPixelConvertParam.nSrcDataLen = pFrame->frameInfo.size;
  283. stPixelConvertParam.nPaddingX = pFrame->frameInfo.paddingX;
  284. stPixelConvertParam.nPaddingY = pFrame->frameInfo.paddingY;
  285. stPixelConvertParam.eBayerDemosaic = demosaicNearestNeighbor;
  286. stPixelConvertParam.eDstPixelFormat = gvspPixelRGB8;
  287. stPixelConvertParam.pDstBuf = pDstBuf;
  288. stPixelConvertParam.nDstBufSize = nDstBufSize;
  289. ret = IMV_PixelConvert(hCam, &stPixelConvertParam);
  290. if(IMV_OK != ret) goto ret_p;
  291. ret = stbi_write_jpg(path, pFrame->frameInfo.width, pFrame->frameInfo.height, 3, pDstBuf, 90);
  292. if(1 == ret) ret = IMV_OK;
  293. else ret = -19;
  294. ret_p:
  295. if(pDstBuf) free(pDstBuf);
  296. return ret;
  297. }
  298. // 保存给定的数据帧到指定格式("BMP"/"JPG")的图像文件,成功返回0值
  299. static int SavePhoto(HANDLE hCam, IMV_Frame *pFrame, DHImgType imgType, const char *imgPath)
  300. {
  301. char *path1 = (char *)imgPath, *path2 = NULL; int ret = IMV_OK; char ext[5];
  302. if(!hCam || !pFrame || !path1 || strlen(path1) <= 0) return -15;
  303. switch(imgType)
  304. {
  305. case IMG_TYPE_BMP: // 保存为.bmp格式
  306. strcpy(ext, ".bmp");
  307. break;
  308. case IMG_TYPE_JPG: // 保存为.jpg格式
  309. strcpy(ext, ".jpg");
  310. break;
  311. default: return -16;
  312. }
  313. if(!xstrcasestr(path1, "right", ext))
  314. {
  315. path2 = (char *)sw_heap_malloc(strlen(path1) + sizeof(ext));
  316. if(path2) sprintf(path2, "%s%s", path1, ext);
  317. else return -17;
  318. if(imgType == IMG_TYPE_BMP) ret = SaveFrameToBmp(hCam, pFrame, path2);
  319. if(imgType == IMG_TYPE_JPG) ret = SaveFrameToJpg(hCam, pFrame, path2);
  320. sw_heap_free(path2);
  321. }
  322. else
  323. {
  324. if(imgType == IMG_TYPE_BMP) ret = SaveFrameToBmp(hCam, pFrame, path1);
  325. if(imgType == IMG_TYPE_JPG) ret = SaveFrameToJpg(hCam, pFrame, path1);
  326. }
  327. return ret;
  328. }
  329. // 获取系统当前USBFS内存大小(MB), 成功返回: >=0, 失败返回: <0值
  330. int DH_GetSysUsbfsMemCurrentSize()
  331. {
  332. const char *usbfsFile = "/sys/module/usbcore/parameters/usbfs_memory_mb";
  333. char buf[MAX_LINE_CHARS]; int val = -1;
  334. FILE *fp = fopen(usbfsFile, "r");
  335. if(fp)
  336. {
  337. if(fgets(buf, sizeof(buf), fp)) sscanf(buf, "%d", &val);
  338. fclose(fp);
  339. }
  340. return val;
  341. }
  342. // 设置系统新的USBFS内存大小(MB), 成功返回: 0值, 失败返回: <0值
  343. int DH_SetSysUsbfsMemSize(int val)
  344. {
  345. const char *usbfsFile = "/sys/module/usbcore/parameters/usbfs_memory_mb";
  346. char buf[MAX_LINE_CHARS]; sprintf(buf, "%d\n", val);
  347. if(val >= 0 && sw_file_update(usbfsFile, "w", buf, strlen(buf)) == strlen(buf)) return 0;
  348. else return -1;
  349. }
  350. // 获取当前已连接的海康相机的数量, 成功返回: >=0, 失败返回: <0值
  351. int DH_GetCameraCount()
  352. {
  353. IMV_DeviceList devList = { 0 };
  354. int ret;
  355. ret = IMV_EnumDevices(&devList, device_type);
  356. if(IMV_OK != ret)
  357. {
  358. sw_log_error("[%s] 枚举相机,执行错误, errCode=%d!!", MODULE_NAME, ret);
  359. return -1;
  360. }
  361. ret = devList.nDevNum;
  362. if(ret > 0)
  363. {
  364. int index = 0, cnt = 0; IMV_DeviceInfo *pDevInfo; char name[MAX_LINE_CHARS];
  365. findp:
  366. pDevInfo = &devList.pDevInfo[index++];
  367. name[0] = '\0';
  368. if(pDevInfo && (pDevInfo->nCameraType == typeU3vCamera || pDevInfo->nCameraType == typeGigeCamera)) strcpy(name, pDevInfo->vendorName);
  369. if(xstrcasecmp(name, manufacturer) == 0 || is_huaray_manu_alias(name)) cnt++;
  370. if(index < devList.nDevNum) goto findp;
  371. else ret = cnt;
  372. }
  373. return ret;
  374. }