struct DV_COMMON_EXPORT product_config {
    int detect_delay_ms = 0;
    int reject_delay_ms = 0;
    int detect_count_oneshot = 1;
    int detect_count = 0;
    int detect_count_interval = 100;
    std::string ref_image_path = "1";
    int cam_x =2;
    int cam_y =3;
    // yuv420p",rgb888
    std::string rpicam_pixel_type = "yuv420p";
    int tmp =44;
};
调试时发现从第一个字符串往后,字段都没有正常初始化
(gdb) info line
Line 1021 of "/repos/dv_app_solution/prism_all/prism/include/prism/prismJson.hpp"
   starts at address 0x7fffeb60f470 <prism::json::fromJsonString<dv_common_config>(std::string const&&)+28>
   and ends at 0x7fffeb60f478 <prism::json::fromJsonString<dv_common_config>(std::string const&&)+36>.
(gdb) list
1016	
1017	template <class T>
1018	static inline std::unique_ptr<T> fromJsonString(const std::string&& str)
1019	{
1020	    std::unique_ptr<T> model = std::make_unique<T>();
1021	    privates::jsonType<T>::type::from_jsonStr(std::move(*model), std::move(str), 0, static_cast<int>(str.length() - 1));
1022	
1023	    return model;
1024	}
1025	template <class T>
(gdb) print *model
$9 = {detect_delay_ms = 0, reject_delay_ms = 0, detect_count_oneshot = 1, detect_count = 0, detect_count_interval = 100, ref_image_path = "", cam_x = 0, 
  cam_y = 0, rpicam_pixel_type = <error reading variable: Cannot create a lazy string with address 0x0, and a non-zero length.>, tmp = 796092265}
(gdb) 
程序会在后面的逻辑中,崩溃在使用 rpicam_pixel_type 这个字段的时候,用 asan 看了是由于这个错误的字符串被认为超过 10 个 T 导致崩溃 随便把它赋值给其他变量就会崩溃
没有思路,有什么方式进一进定位吗
解决问题了,来汇报一下
我自己写的反射是非侵入式的,我把反射的元数据和结构体分为两个头文件 正常在dv_common动态库外是不应该使用元数据的类的, 但由于配置文件是一个大类,没有拆得很清楚, dv_tools里有一个通讯用类也用元数据头文件进行dv_common_config的配置类json序列化了,所以也有特化符号 (正常为了防止冗余,应该只在dv_common里使用反射元数据,是我没设计好
libdv_tools是一个比较底层的工具动态库,所以我平时都不编译它, 这次在配置文件中加了两个字段 ,没有为dv_tools重新编译特化符号, 又因为dv_tools加载比较早,程序应该是从dv_tools取用旧的反射元数据符号了,导致问题
#dv_tools 3月14编译的
root@a3beb7022bce:/repos/dv_app_solution# ll bin/arm64/
-rwxr-xr-x  1 root       root  34452552 Apr 11 03:07 libdv_common.so
-rwxr-xr-x  1 root       root  24991736 Mar 14 02:10 libdv_tools.so
#程序模块加载配置
{
    "version":"1.0.0.0",
    "plugins":[
        "prism_qt_core",
        "prism_qt_ui",
        "dv_tools",       //在dv_common之前加载
        "dv_common",
        "dv_algorithm",
        "dv_camera"
    ],
    "openGLVersion":"default",
    "usingGUI":true
}
重新编译 dv_tools后部署后就正常了
下周再重构一下,把dv_common_config的序列化放到dv_common里,向外导出
|      1billlee      204 天前 DV_COMMON_EXPORT 是什么? | 
|      2shapper      204 天前 DV_COMMON_EXPORT 估计是到处宏; 如果导出的模块,结构的所有 std::string 全部要改成 char*,不然以后跨模块,有得你受咯 | 
|      3yanlx      204 天前 用 gbd watch 看变量在哪里被修改了。 | 
|      5OBJECTION      204 天前 丢给 gpt 阔以直接问出来啊。fromJsonString 这玩意是直接内存拷贝啊。。string 是类啊。。关键其实不在于你上面的定义。。 而是感觉你 guide 看漏了。。 | 
|  |      6ysc3839      204 天前 via Android from_jsonStr 的代码呢? | 
|  |      7yolee599      204 天前 via Android 有点看不懂,现在的 c++ 已经可以在定义结构体的时候赋值了吗?建议贴代码出来 | 
|  |      8geligaoli      204 天前  1 2 楼说的对,如果是 export 的,不要使用 c++类,只使用基本数据类型,你无法知道调用者用的是哪种 stl 的实现和内存分配,不一致就会出现莫名其妙的错误。 | 
|  |      9ksedz      204 天前 是不是因为你给了短字符串,编译器错误优化到了栈上? | 
|      10cnbatch      204 天前 | 
|  |      11jujusama      204 天前 ```cpp std::unique_ptr<T> model = std::make_unique<T>(); func(std::move(*model)); return model; ``` 这 model 不是被 move 了吗?这是在干嘛 | 
|      12yuzii      204 天前 导出动态库、std::string 、内存分配问题 这 3 个结合大概率是 c++ abi 混用引起的吧 | 
|      14GeruzoniAnsasu      204 天前 fromJsonString 的入参是右值引用,会调用对象的移动构造 你的 product_config 没有 explict 的移动构造,那么会根据内部成员的移动构造生成隐式移动构造 调用完移动构造后 std::string 会变为无效状态 在你这个地方报错显示 std::string 成员的指针已经变成了 null ,说明之前被移动过了,合理推测是你使用了类似 product_config c; a = fromJsonString(c); b = fromJsonString(c); 的代码 | 
|      15hwdq0012 OP @GeruzoniAnsasu  @jujusama 我断点在 make_unique 的那一行,hit 后 next 执行一步,那时在 gdb 里 print *model ,就已经是这样了,应该不是移动构造导致的,因为单元测试过嵌套类 再者,是第一个字符串后面的 int 也有问题了, 前面的 Int 没有问题 @yuzii 桌面软件的动态库,一起编译的,只在桌面 app 上用,所以应该不是 abi 问题,像是内存问题 | 
|  |      16lixile      204 天前 asan 扫过了  那 ubsan 扫过了么 如果调用方有多线程 tsan 扫过了么 | 
|      19yanlx      204 天前 https://godbolt.org/z/WeqKM6qr3 做了个最小例子,看起来各家编译器编译结果都没啥问题。还是 DV_COMMON_EXPORT 和 from_jsonStr 嫌疑比较大。 另外你的 gdb 里面 ref_image_path 看起来也不对劲。 | 
|  |      20bfjm      204 天前 via iPhone 多线程问题吧  这块内存已经被破坏掉了 | 
|  |      21exch4nge      204 天前 gdb 显示的信息里 address 是 prism::json::fromJsonString<dv_common_config> 所以 dv_common_config 是什么?是 product_config 吗? | 
|      22hwdq0012 OP | 
|  |      24geelaw      204 天前 @hwdq0012 #22 代码里 product_config 没有基类。 楼主的代码疑点很多,比如 const && 是无法被移走的,几乎所有代码里 const && 都是错误。 @GeruzoniAnsasu #14 std::move 并不会导致对象变化,因为只是 static_cast 到 rvalue reference 而已。 被移动过的 std::string 依然处于有效状态,调试器显示 rpicam_pixel_type 是 address 0 length positive 所以那个位置是无效状态,因此必然存在 bug ,而且并不是由正常的移动构造引起的。 | 
|      25cxiaobao      204 天前 template <class T> 1018 static inline std::unique_ptr<T> fromJsonString(const std::string&& str) 1019 { 1020 std::unique_ptr<T> model = std::make_unique<T>(); 1021 privates::jsonType<T>::type::from_jsonStr(std::move(*model), std::move(str), 0, static_cast<int>(str.length() - 1)); 1022 ~~~~~~~~~~~~~~ model 是移动语义进入 from_jsonStr 方法的,如果 from_jsonStr 取走了数据所有权,model 会被清空 1023 return model; ~~~~~~~~~ 这里不应该返回 model ,因为 model 可能已经被清空了, 通常,from_jsonStr 应该返回一个对象,这个对象取得了原本 model 中的数据,应该考虑返回这个对象 1024 } | 
|  |      26liberize      203 天前 via Android 检查一下这个头文件是不是有 2 个不同的版本 | 
|      27hwdq0012 OP | 
|      28hwdq0012 OP | 
|  |      29lixile      200 天前 `dv_tools 里有一个通讯用类也用元数据头文件进行 dv_common_config 的配置类 json 序列化了` 我不知道我有没有理解错 居然没有用 cmake 之类的 对这个配置文件 作为 dv_tools 的依赖项么 这样只要配置文件发生变更 就会自动增量重新编译 还是说这几个项目没有关联在一起 而是独立的 repo 且没有用一个 cmake 工程整体管理? @hwdq0012 |