Portál AbcLinuxu, 5. května 2025 21:30
Snažím se zabudovat do aplikace knihovnu Qt Charts, ale mám problém s úniky paměti.
Používám AddressSanitizer (ASan) pro odhalování mých chyb. I bez těch grafů mi to hlásilo nějaké úniky, ale přišlo mi, že by to snad neměla být moje chyba, tak jsem to dal ignorovat -- souborem asan-ignore.txt
:
leak:/lib/x86_64-linux-gnu/libdbus-1.so leak:/usr/lib/x86_64-linux-gnu/libfontconfig.so
A program pak pouštím s LSAN_OPTIONS=suppressions=asan-ignore.txt
. Je to takhle OK (resp. chyba v knihovně) nebo dělám něco špatně? (hlásilo mi to u triviální aplikace s prázdným oknem, tak to spíš tipuji na chybu v knihovně)
Když udělám záměrnou chybu ve svém kódu, tak mi ji ASan hlásí.
Situace se ale výrazně zhoršila, když jsem do toho přidal ty Qt Charts. Jednak nevím, jestli to nejsou moje chyby, a jednak jsou na místech, která kdybych ignoroval, tak to asi skryje i potenciální moje chyby, což nechci.
================================================================= ==1030==ERROR: LeakSanitizer: detected memory leaks Direct leak of 6656 byte(s) in 26 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46cd0b58ed (/usr/lib/x86_64-linux-gnu/libfontconfig.so.1+0x1d8ed) Indirect leak of 8478 byte(s) in 10 object(s) allocated from: #0 0x7f46d81e6f40 in realloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdef40) #1 0x7f46ca49ecfc (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2ecfc) Indirect leak of 6048 byte(s) in 84 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d70ac441 (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0x2bc441) Indirect leak of 5830 byte(s) in 29 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46d6e9d8c1 in QArrayData::allocate(unsigned long, unsigned long, unsigned long, QFlags<QArrayData::AllocationOption>) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0xad8c1) Indirect leak of 2240 byte(s) in 20 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d55cb581 in QBrush::init(QColor const&, Qt::BrushStyle) (/usr/lib/x86_64-linux-gnu/libQt5Gui.so.5+0x2bb581) Indirect leak of 2128 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d54e8ed3 in QFontDatabase::load(QFontPrivate const*, int) (/usr/lib/x86_64-linux-gnu/libQt5Gui.so.5+0x1d8ed3) Indirect leak of 1944 byte(s) in 27 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d70ad6f2 in QObjectPrivate::connectImpl(QObject const*, int, QObject const*, void**, QtPrivate::QSlotObjectBase*, Qt::ConnectionType, int const*, QMetaObject const*) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0x2bd6f2) Indirect leak of 1904 byte(s) in 17 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d70b0e01 in QObject::QObject(QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0x2c0e01) Indirect leak of 1632 byte(s) in 51 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46cd0b5fd8 (/usr/lib/x86_64-linux-gnu/libfontconfig.so.1+0x1dfd8) Indirect leak of 808 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca49eacd in _dbus_mem_pool_alloc (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2eacd) Indirect leak of 576 byte(s) in 8 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d56c0976 in QPen::QPen(QColor const&) (/usr/lib/x86_64-linux-gnu/libQt5Gui.so.5+0x3b0976) #2 0x633f418b30ad74ff (<unknown module>) Indirect leak of 528 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca4805fd (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x105fd) Indirect leak of 519 byte(s) in 50 object(s) allocated from: #0 0x7f46d817f538 in strdup (/usr/lib/x86_64-linux-gnu/libasan.so.4+0x77538) #1 0x7f46cd0b52f4 in FcValueSave (/usr/lib/x86_64-linux-gnu/libfontconfig.so.1+0x1d2f4) Indirect leak of 512 byte(s) in 8 object(s) allocated from: #0 0x7f46d81e6f40 in realloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdef40) #1 0x7f46d6eee2e0 in QListData::realloc_grow(int) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0xfe2e0) Indirect leak of 512 byte(s) in 16 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d70ab961 in QObjectPrivate::addConnection(int, QObjectPrivate::Connection*) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0x2bb961) Indirect leak of 480 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca47d9e7 (/lib/x86_64-linux-gnu/libdbus-1.so.3+0xd9e7) #2 0x7ffffffff (<unknown module>) Indirect leak of 480 byte(s) in 30 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e56afe (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xceafe) Indirect leak of 464 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca49ab5f (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2ab5f) Indirect leak of 448 byte(s) in 8 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46ca4852dd in _dbus_credentials_new (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x152dd) Indirect leak of 280 byte(s) in 7 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46ca4a4f01 (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x34f01) Indirect leak of 272 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e22a03 in QtCharts::QBarCategoryAxis::QBarCategoryAxis(QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0x9aa03) Indirect leak of 264 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e1ff83 in QtCharts::QValueAxis::QValueAxis(QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0x97f83) Indirect leak of 264 byte(s) in 3 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca493aba (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x23aba) Indirect leak of 224 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca49babb in _dbus_hash_table_new (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2babb) Indirect leak of 192 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d54bf435 in QFont::detach() (/usr/lib/x86_64-linux-gnu/libQt5Gui.so.5+0x1af435) Indirect leak of 192 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca49069f in _dbus_message_loader_new (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2069f) Indirect leak of 192 byte(s) in 4 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46ca4a5021 (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x35021) Indirect leak of 160 byte(s) in 4 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46ca4a4eaa (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x34eaa) Indirect leak of 160 byte(s) in 5 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46d6eee131 in QListData::detach_grow(int*, int) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0xfe131) Indirect leak of 160 byte(s) in 5 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e52871 (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xca871) Indirect leak of 160 byte(s) in 5 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e527e5 (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xca7e5) Indirect leak of 160 byte(s) in 5 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e5274b (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xca74b) Indirect leak of 160 byte(s) in 5 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e526a3 (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xca6a3) #2 0x7f46d6471b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 152 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e5a953 in QtCharts::QStackedBarSeries::QStackedBarSeries(QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xd2953) #2 0x7f46d6471b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 128 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6f40 in realloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdef40) #1 0x7f46ca49097d in _dbus_message_loader_get_unix_fds (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2097d) #2 0x633f418b30ad74ff (<unknown module>) Indirect leak of 128 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca492477 (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x22477) Indirect leak of 112 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7de9bc1 (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0x61bc1) #2 0x60300004e75f (<unknown module>) Indirect leak of 96 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca4805c8 (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x105c8) Indirect leak of 96 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca4805b7 (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x105b7) Indirect leak of 96 byte(s) in 3 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46d6eee1eb in QListData::detach(int) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0xfe1eb) Indirect leak of 96 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d54bd2e5 in QFont::QFont(QString const&, int, int, bool) (/usr/lib/x86_64-linux-gnu/libQt5Gui.so.5+0x1ad2e5) Indirect leak of 88 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e55c3d in QtCharts::QBarSet::QBarSet(QString, QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xcdc3d) #2 0xf79e43f4 (<unknown module>) Indirect leak of 88 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e55c3d in QtCharts::QBarSet::QBarSet(QString, QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xcdc3d) Indirect leak of 88 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e55c3d in QtCharts::QBarSet::QBarSet(QString, QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xcdc3d) #2 0x7ffe6738b98f (<unknown module>) Indirect leak of 88 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e55c3d in QtCharts::QBarSet::QBarSet(QString, QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xcdc3d) #2 0x7f46d92c6147 (<unknown module>) Indirect leak of 88 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7e55c3d in QtCharts::QBarSet::QBarSet(QString, QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xcdc3d) #2 0x7f46d813372e (/usr/lib/x86_64-linux-gnu/libasan.so.4+0x2b72e) Indirect leak of 82 byte(s) in 4 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46ca49caae in _dbus_strdup (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2caae) Indirect leak of 80 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca49e9b3 in _dbus_mem_pool_new (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2e9b3) Indirect leak of 70 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46ca49f209 in _dbus_string_copy_data (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2f209) #2 0x633f418b30ad74ff (<unknown module>) Indirect leak of 64 byte(s) in 8 object(s) allocated from: #0 0x7f46d81e6b50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f46ca49ef43 (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2ef43) Indirect leak of 48 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca492852 (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x22852) Indirect leak of 48 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6d38 in __interceptor_calloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xded38) #1 0x7f46ca47e591 (/lib/x86_64-linux-gnu/libdbus-1.so.3+0xe591) Indirect leak of 32 byte(s) in 2 object(s) allocated from: #0 0x7f46d81e6f40 in realloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdef40) #1 0x7f46ca49b901 (/lib/x86_64-linux-gnu/libdbus-1.so.3+0x2b901) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x5617ab52e845 in main ../stackedbarchart/main.cpp:59 #2 0x7f46d6471b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x5617ab52dab4 in main ../stackedbarchart/main.cpp:49 #2 0x7f46d6471b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f46d7deafe6 (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0x62fe6) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x5617ab52da31 in main ../stackedbarchart/main.cpp:48 #2 0x7f46d6471b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x5617ab52d9ae in main ../stackedbarchart/main.cpp:47 #2 0x7f46d6471b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x5617ab52ecf3 in main ../stackedbarchart/main.cpp:77 #2 0x7f46d6471b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x5617ab52d92b in main ../stackedbarchart/main.cpp:46 #2 0x7f46d6471b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f46d81e8458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x5617ab52d8a8 in main ../stackedbarchart/main.cpp:45 #2 0x7f46d6471b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) SUMMARY: AddressSanitizer: 46987 byte(s) leaked in 494 allocation(s).
Zkoušel jsem přidat ASan i k oficiálnímu příkladu, který je součástí Qt, a taky to hlásí chyby.
Soubor stackedbarchart.pro
:
QT += charts CONFIG += sanitizer sanitize_address SOURCES += \ main.cpp target.path = $$[QT_INSTALL_EXAMPLES]/charts/stackedbarchart INSTALLS += target
Soubor main.cpp
:
/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the Qt Charts module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:GPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 or (at your option) any later version ** approved by the KDE Free Qt Foundation. The licenses are as published by ** the Free Software Foundation and appearing in the file LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include <QtWidgets/QApplication> #include <QtWidgets/QMainWindow> #include <QtCharts/QChartView> #include <QtCharts/QStackedBarSeries> #include <QtCharts/QBarSet> #include <QtCharts/QLegend> #include <QtCharts/QBarCategoryAxis> QT_CHARTS_USE_NAMESPACE int main(int argc, char *argv[]) { QApplication a(argc, argv); //![1] QBarSet *set0 = new QBarSet("Jane"); QBarSet *set1 = new QBarSet("John"); QBarSet *set2 = new QBarSet("Axel"); QBarSet *set3 = new QBarSet("Mary"); QBarSet *set4 = new QBarSet("Samantha"); *set0 << 1 << 2 << 3 << 4 << 5 << 6; *set1 << 5 << 0 << 0 << 4 << 0 << 7; *set2 << 3 << 5 << 8 << 13 << 8 << 5; *set3 << 5 << 6 << 7 << 3 << 4 << 5; *set4 << 9 << 7 << 5 << 3 << 1 << 2; //![1] //![2] QStackedBarSeries *series = new QStackedBarSeries(); series->append(set0); series->append(set1); series->append(set2); series->append(set3); series->append(set4); //![2] //![3] QChart *chart = new QChart(); chart->addSeries(series); chart->setTitle("Simple stackedbarchart example"); chart->setAnimationOptions(QChart::SeriesAnimations); //![3] //![4] QStringList categories; categories << "Jan" << "Feb" << "Mar" << "Apr" << "May" << "Jun"; QBarCategoryAxis *axis = new QBarCategoryAxis(); axis->append(categories); chart->createDefaultAxes(); chart->setAxisX(axis, series); //![4] //![5] chart->legend()->setVisible(true); chart->legend()->setAlignment(Qt::AlignBottom); //![5] //![6] QChartView *chartView = new QChartView(chart); chartView->setRenderHint(QPainter::Antialiasing); //![6] //![7] QMainWindow window; window.setCentralWidget(chartView); window.resize(420, 300); window.show(); //![7] return a.exec(); }
Qt knihovna by to přece měla všechno uklidit na základě hierarchie (parent) ne? Zkoušel jsem s tím různě laborovat, přidávat i explicitní delete
do destruktoru své třídy, ale to nepomohlo, maximálně to pak někdy hlásilo heap-use-after-free
, když jsem smazal něco, co se ještě používalo. Ten můj program (trochu složitější něž příklad výše -- mám tam model a QVBarModelMapper
) se mi podařilo dostat do stavu, kdy to chyby hlásí jen někdy -- asi je tam nějaký souběh a záleží, kdy okno zavřu a jestli se stihne vše uklidit nebo ne (?). Ale to je celkem jedno -- potřeboval bych opravit aspoň ten příklad výše.
Je chyba v tom kódu nebo v knihovně? A pokud v knihovně, dá se to zatím (nahlásil bych to u Qt) nějak ignorovat, aniž by to pak ignorovalo i moje chyby?
P.S. Našel jsem hlášenou chybu QTBUG-58802, která by měla být opravená v Qt 5.9.0 (já mám Qt 5.9.5).
Přepiš to do Pythonu, nebudeš muset takové věci řešit. C++ jsem měl pár semestrů ve škole a nešlo mi to.Můžu potvrdit, že po několika letech kódění UI v C++/Qt jsem taky dospěl k přesvědčení, že C++ je fajn jazyk na backedové věci ale UI se v něm tvoří s elegancí tančícího slona a ani Qt to tolik neusnadňuje. Kdybys měl zájem, mrkni se na PySide2 či PyQt. Nicméně k věci: Ty leaky, co pozoruješ nemusí být nutně leaky per se ale nějaká sdílená paměť. Qtčka se snaží různé věci cacheovat a tyhle cache se můžou v různých analyzátorech objevit jako memory leaky. Píšeš-li to v QtCreatoru, zkus to prohnat Valgrindem
(Analyze -> Valgrind memory analyzer
) a v nastavení výstupu (ikona nálevky na "meziliště" zruš "External errors". Jinak pro vykreslování různých grafů existuje třeba i knihovna QwtPlot se kterou jsem zatím žádné problémy neměl.
Píšeš-li to v QtCreatoru, zkus to prohnat Valgrindem (Analyze -> Valgrind memory analyzer) a v nastavení výstupu (ikona nálevky na "meziliště" zruš "External errors".
Dík, líbí se mi, že je to součást IDE. Ale není lepší ten ASan? Vagrind mi přišel jednak hrozně pomalý a jednak pořádně ani nenajde tu moji zkušební chybu:
char* chyba = new char[66600];
Resp. v základním výpisu na příkazové řádce jsem to neviděl a když jsem dal valgrind --leak-check=full
, tak to zase vysypalo tolik chyb, že se to tam špatně hledalo (ale vypsaná tam už byla). A v tom QtCreatoru se mi zobrazuje právě až v těch Externích chybách -- když je odškrtnu, tak tam pro změnu nejsou chyby žádné.
qDebug()
.
To už to rovnou můžu psát v Javě :-)
Jenže ten Python (nebo Java nebo jiný jazyk) k tomu přidá akorát další vrstvu, ale pod tím bude to samé Qt a C++, takže to tenhle problém neodstraní.
A co se týče chyb v mém vlastním kódu -- C++ s RAII, nějaké ty chytré ukazatele a/nebo Qt mi připadá celkem použitelné. Není to sice tak pohodlné jako jazyk s GC a člověk nad tím musí trochu víc přemýšlet, ale IMHO by v tom měly jít psát spolehlivé aplikace, kde neuniká paměť.
No so spolužiakom sme to minule riešili a ja som mu poradil prepísať to do C++ pretože PySide neskutočne leakuje. Napr. tento malý programík u mňa leakuje 2MB/s.
Leakuje ti tam ten QChart
a všechno co je v něm.
w.setCentralWidget(chartView)
. Tím se přenese vlastnictví chartView
na hlavní okno, které je vytvořeno na zásobníku a když main()
doběhne, automaticky se zavolá destruktor, který by měl zničit i ten chartView
. Když ten demokód proženu Valgrindem, žádné úniky nehlásí. Přidám-li tam někam int *x = new int{65}
, pak teprve dostanu od Valgrind varování o leaku. Obecně platí, že výstupy z nástrojů jako ASan či Valgrind je třeba umět interpretovat, jinak může člověk strávit hodiny pronásledováním neexistujících problémů.
Píšu QChart, ne QChartView. QChartView je dealokováno na konci QMainWindow, ale je to potomek toho QChart a potomci své rodiče v Qt neruší. A jinde se ten QChart neruší.
QChart
sám o sobě není QWidget
a v tom kódu se nejdříve vyrobí bez rodiče. QChartView
už QWidget je. Konstruktor, kterým se v tom kódu vytváří je tento: QChartView::QChartView(QChart *chart, QWidget *parent = nullptr)
. chart
je pak potomek chartView
a chartView
zase potomek window
. Kdyby to mělo fungovat tak, jak jsi naznačoval, docházelo by tam k docela ošklivému reparentingu.
Jestli máš nainstalované QtCharts, schválně zkus nějakým analyzátorem prohnat toto:
#include <QtWidgets/QApplication> #include <QtWidgets/QMainWindow> #include <QtCharts/QChartView> #include <QtCharts/QStackedBarSeries> #include <QtCharts/QBarSet> #include <QtCharts/QLegend> #include <QtCharts/QBarCategoryAxis> QT_CHARTS_USE_NAMESPACE static int disp(QApplication &a, QChartView *view) { QMainWindow window; window.setCentralWidget(view); window.resize(420, 300); window.show(); return a.exec(); } int main(int argc, char *argv[]) { QApplication a(argc, argv); QBarSet *set0 = new QBarSet("Jane"); QBarSet *set1 = new QBarSet("John"); QBarSet *set2 = new QBarSet("Axel"); QBarSet *set3 = new QBarSet("Mary"); QBarSet *set4 = new QBarSet("Samantha"); *set0 << 1 << 2 << 3 << 4 << 5 << 6; *set1 << 5 << 0 << 0 << 4 << 0 << 7; *set2 << 3 << 5 << 8 << 13 << 8 << 5; *set3 << 5 << 6 << 7 << 3 << 4 << 5; *set4 << 9 << 7 << 5 << 3 << 1 << 2; QStackedBarSeries *series = new QStackedBarSeries(); series->append(set0); series->append(set1); series->append(set2); series->append(set3); series->append(set4); QChart *chart = new QChart(); chart->addSeries(series); chart->setTitle("Simple stackedbarchart example"); chart->setAnimationOptions(QChart::SeriesAnimations); QStringList categories; categories << "Jan" << "Feb" << "Mar" << "Apr" << "May" << "Jun"; QBarCategoryAxis *axis = new QBarCategoryAxis(); axis->append(categories); chart->createDefaultAxes(); chart->setAxisX(axis, series); chart->legend()->setVisible(true); chart->legend()->setAlignment(Qt::AlignBottom); QChartView *chartView = new QChartView(chart); chartView->setRenderHint(QPainter::Antialiasing); int ret = disp(a, chartView); delete chart; return ret; }Od Valgrindu bys měl na konci dostat "invalid read", bez toho delete to bude všechno OK.
Hmm, QChartView se opravdu chová jinak, než jsem předpokládal (čekal jsem, že je to stejný jako u QGraphicsView, který scénu co zobrazuje nereparentuje). Zas tak detailně jsem ten příklad nezkoumal a už vůbec jsem to nepouštěl ve Valgrindu, tolik volnýho času zase nemám
ASan vyhodí ještě hezčí výstup :-)
================================================================= ==9639==ERROR: AddressSanitizer: heap-use-after-free on address 0x60600005f9c0 at pc 0x555c34823fc4 bp 0x7ffd5b1f4230 sp 0x7ffd5b1f4220 READ of size 8 at 0x60600005f9c0 thread T0 #0 0x555c34823fc3 in main ../stackedbarchart/main.cpp:105 #1 0x7f1523b49b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) #2 0x555c34822259 in _start (~/src/qt/examples/charts/build-stackedbarchart-Desktop-Ladu011bnu00ed/stackedbarchart+0x2259) 0x60600005f9c0 is located 0 bytes inside of 56-byte region [0x60600005f9c0,0x60600005f9f8) freed by thread T0 here: #0 0x7f15258c19d8 in operator delete(void*, unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe19d8) #1 0x7f152506bfdd in QGraphicsScene::clear() (/usr/lib/x86_64-linux-gnu/libQt5Widgets.so.5+0x453fdd) previously allocated by thread T0 here: #0 0x7f15258c0458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x555c3482390d in main ../stackedbarchart/main.cpp:68 #2 0x7f1523b49b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) SUMMARY: AddressSanitizer: heap-use-after-free ../stackedbarchart/main.cpp:105 in main Shadow bytes around the buggy address: 0x0c0c80003ee0: fa fa fa fa 00 00 00 00 00 00 00 00 fa fa fa fa 0x0c0c80003ef0: 00 00 00 00 00 00 00 00 fa fa fa fa 00 00 00 00 0x0c0c80003f00: 00 00 00 00 fa fa fa fa 00 00 00 00 00 00 00 00 0x0c0c80003f10: fa fa fa fa 00 00 00 00 00 00 00 00 fa fa fa fa 0x0c0c80003f20: 00 00 00 00 00 00 00 00 fa fa fa fa 00 00 00 00 =>0x0c0c80003f30: 00 00 00 00 fa fa fa fa[fd]fd fd fd fd fd fd fa 0x0c0c80003f40: fa fa fa fa fd fd fd fd fd fd fd fd fa fa fa fa 0x0c0c80003f50: fd fd fd fd fd fd fd fd fa fa fa fa fd fd fd fd 0x0c0c80003f60: fd fd fd fd fa fa fa fa fd fd fd fd fd fd fd fd 0x0c0c80003f70: fa fa fa fa fd fd fd fd fd fd fd fd fa fa fa fa 0x0c0c80003f80: fd fd fd fd fd fd fd fd fa fa fa fa fd fd fd fd Shadow byte legend (one shadow byte represents 8 application bytes): Addressable: 00 Partially addressable: 01 02 03 04 05 06 07 Heap left redzone: fa Freed heap region: fd Stack left redzone: f1 Stack mid redzone: f2 Stack right redzone: f3 Stack after return: f5 Stack use after scope: f8 Global redzone: f9 Global init order: f6 Poisoned by user: f7 Container overflow: fc Array cookie: ac Intra object redzone: bb ASan internal: fe Left alloca redzone: ca Right alloca redzone: cb ==9639==ABORTING
Tohle jsem právě už zkoušel, různě mazat, ale dopadlo to leda tak, že se něco mazalo dvakrát resp. používalo po smazání, ale úniky paměti to stejně neodstranilo.
Obecně platí, že výstupy z nástrojů jako ASan či Valgrind je třeba umět interpretovat, jinak může člověk strávit hodiny pronásledováním neexistujících problémů.
1) Mám trochu problém s tím, že v té knihovně možná je chyba, která by se při dlouhodobějším používání projevila. U té mojí aplikace je to jedno, tam se vykreslí pár grafů a pak to uživatel zavře, takže to žádný problém nezpůsobí. Ale co kdyby to bylo něco jako webový prohlížeč nebo desktopové prostředí? To běží týdny nebo měsíce a uživatel tam pořád něco otvírá a zavírá -- tam by se ty úniky paměti nastřádaly a problém by to byl. Tohle přece musí být nějak vyřešené, ne? <javaTrolling>Nebo je je jediným řešením garbage collector? :-)</javaTrolling>
2) Chápu, že je potřeba se to naučit číst a chápu i že ne vše je dokonalé a nějaké menší chyby nebo plané poplachy v těch knihovnách být můžou. Můj plán tedy byl, že do LSAN_OPTIONS=suppressions=
přidám (zatím tam mám libdbus-1.so
a libfontconfig.so
) ignorování známých chyb v knihovnách, a pak mi AddressSanitizer bude hlásit už jen moje chyby. Jenže jak na to -- co tam mám přidat, když mi hlásí tohle?
$ LSAN_OPTIONS=suppressions=stackedbarchart/asan-ignore.txt build-stackedbarchart-Desktop-Ladu011bnu00ed/stackedbarchart ================================================================= ==10398==ERROR: LeakSanitizer: detected memory leaks Indirect leak of 6048 byte(s) in 84 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717254441 (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0x2bc441) Indirect leak of 5830 byte(s) in 29 object(s) allocated from: #0 0x7f871838eb50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f87170458c1 in QArrayData::allocate(unsigned long, unsigned long, unsigned long, QFlags<QArrayData::AllocationOption>) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0xad8c1) Indirect leak of 2240 byte(s) in 20 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8715773581 in QBrush::init(QColor const&, Qt::BrushStyle) (/usr/lib/x86_64-linux-gnu/libQt5Gui.so.5+0x2bb581) Indirect leak of 2128 byte(s) in 2 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8715690ed3 in QFontDatabase::load(QFontPrivate const*, int) (/usr/lib/x86_64-linux-gnu/libQt5Gui.so.5+0x1d8ed3) Indirect leak of 1944 byte(s) in 27 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f87172556f2 in QObjectPrivate::connectImpl(QObject const*, int, QObject const*, void**, QtPrivate::QSlotObjectBase*, Qt::ConnectionType, int const*, QMetaObject const*) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0x2bd6f2) Indirect leak of 1904 byte(s) in 17 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717258e01 in QObject::QObject(QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0x2c0e01) Indirect leak of 576 byte(s) in 8 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8715868976 in QPen::QPen(QColor const&) (/usr/lib/x86_64-linux-gnu/libQt5Gui.so.5+0x3b0976) #2 0x29183d753b2edeff (<unknown module>) Indirect leak of 512 byte(s) in 8 object(s) allocated from: #0 0x7f871838ef40 in realloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdef40) #1 0x7f87170962e0 in QListData::realloc_grow(int) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0xfe2e0) Indirect leak of 512 byte(s) in 16 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717253961 in QObjectPrivate::addConnection(int, QObjectPrivate::Connection*) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0x2bb961) Indirect leak of 480 byte(s) in 30 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717ffeafe (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xceafe) Indirect leak of 272 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717fcaa03 in QtCharts::QBarCategoryAxis::QBarCategoryAxis(QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0x9aa03) Indirect leak of 264 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717fc7f83 in QtCharts::QValueAxis::QValueAxis(QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0x97f83) Indirect leak of 192 byte(s) in 2 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8715667435 in QFont::detach() (/usr/lib/x86_64-linux-gnu/libQt5Gui.so.5+0x1af435) Indirect leak of 160 byte(s) in 5 object(s) allocated from: #0 0x7f871838eb50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f8717096131 in QListData::detach_grow(int*, int) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0xfe131) Indirect leak of 160 byte(s) in 5 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717ffa871 (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xca871) Indirect leak of 160 byte(s) in 5 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717ffa7e5 (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xca7e5) Indirect leak of 160 byte(s) in 5 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717ffa74b (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xca74b) Indirect leak of 160 byte(s) in 5 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717ffa6a3 (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xca6a3) #2 0x7f8716619b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 152 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8718002953 in QtCharts::QStackedBarSeries::QStackedBarSeries(QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xd2953) #2 0x7f8716619b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 112 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717f91bc1 (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0x61bc1) #2 0x60300004e2df (<unknown module>) Indirect leak of 96 byte(s) in 3 object(s) allocated from: #0 0x7f871838eb50 in __interceptor_malloc (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xdeb50) #1 0x7f87170961eb in QListData::detach(int) (/usr/lib/x86_64-linux-gnu/libQt5Core.so.5+0xfe1eb) Indirect leak of 96 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f87156652e5 in QFont::QFont(QString const&, int, int, bool) (/usr/lib/x86_64-linux-gnu/libQt5Gui.so.5+0x1ad2e5) Indirect leak of 88 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717ffdc3d in QtCharts::QBarSet::QBarSet(QString, QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xcdc3d) #2 0xeae59bd6 (<unknown module>) Indirect leak of 88 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717ffdc3d in QtCharts::QBarSet::QBarSet(QString, QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xcdc3d) Indirect leak of 88 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717ffdc3d in QtCharts::QBarSet::QBarSet(QString, QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xcdc3d) #2 0x7fff1428e1cf (<unknown module>) Indirect leak of 88 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717ffdc3d in QtCharts::QBarSet::QBarSet(QString, QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xcdc3d) #2 0x7f871946d147 (<unknown module>) Indirect leak of 88 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717ffdc3d in QtCharts::QBarSet::QBarSet(QString, QObject*) (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0xcdc3d) #2 0x7f87182db72e (/usr/lib/x86_64-linux-gnu/libasan.so.4+0x2b72e) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x55dbc98ba845 in main ../stackedbarchart/main.cpp:59 #2 0x7f8716619b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x55dbc98b9ab4 in main ../stackedbarchart/main.cpp:49 #2 0x7f8716619b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x7f8717f92fe6 (/usr/lib/x86_64-linux-gnu/libQt5Charts.so.5+0x62fe6) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x55dbc98b9a31 in main ../stackedbarchart/main.cpp:48 #2 0x7f8716619b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x55dbc98b99ae in main ../stackedbarchart/main.cpp:47 #2 0x7f8716619b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x55dbc98bacf3 in main ../stackedbarchart/main.cpp:77 #2 0x7f8716619b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x55dbc98b992b in main ../stackedbarchart/main.cpp:46 #2 0x7f8716619b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) Indirect leak of 24 byte(s) in 1 object(s) allocated from: #0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x55dbc98b98a8 in main ../stackedbarchart/main.cpp:45 #2 0x7f8716619b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96) ----------------------------------------------------- Suppressions used: count bytes template 78 13390 /lib/x86_64-linux-gnu/libdbus-1.so 127 8807 /usr/lib/x86_64-linux-gnu/libfontconfig.so ----------------------------------------------------- SUMMARY: AddressSanitizer: 24790 byte(s) leaked in 289 allocation(s).
Čeho se tam můžu chytnout, když tam jsou řádky jako tyhle?
#0 0x7f8718390458 in operator new(unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0458) #1 0x55dbc98b98a8 in main ../stackedbarchart/main.cpp:45 #2 0x7f8716619b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96)
Jde mi o to, abych toho neignoroval moc a nezakrylo to moje vlastní chyby. Když tam přidám primitivní chybu:
char* chyba = new char[666]; return returnCode;
Tak to způsobí hlášku o přímém úniku:
Direct leak of 666 byte(s) in 1 object(s) allocated from: #0 0x7fd512c20618 in operator new[](unsigned long) (/usr/lib/x86_64-linux-gnu/libasan.so.4+0xe0618) #1 0x558aed43af84 in main ../stackedbarchart/main.cpp:116 #2 0x7fd510ea9b96 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21b96)
To tedy chceš říct, že mám ignorovat všechny nepřímé úniky? Nehrozí (alespoň v Qt), že bych takový nepřímý únik způsobil chybou ve svém kódu?
Máš tip na nějakou knihovnu pro grafy v GTK? Nemusí to být tak velké jako Qt Charts, stačí základní "byznysové" grafy (koláčové, sloupcové...).
Myslím, že tu nie je žiaden memory leak. Ešte pred tým než sa pustím do mojej použitej metódy detekcie leakov by som chcel napísať pár všeobecných tvrdení o leakoch.
Memory leak je situácia kedy pamäť ostáva alokovaná, ale už nie je prístupná programátorovi (stratil na ňu referenciu). Detekcia memory leakov v nízkoúrovňových jazykoch je veľmi náročná pretože musí byť prekontrolované, či sú dostupné všetky alokované bloky (teda existuje pointer na začiatok alokovaného bloku pamäte). Problém je, že pointer nemusí vždy smerovať na začiatok a môže byť pre leak detektor akože stratený aj keď v skutočnosti je to nejaký posuvný pointer, ktorý pozná svoju pozíciu. To je jeden z dôvodov prečo môže vyhadzovať false positive. Taktiež bude vyhadzovať false positive ak pri ukončení aplikácie zostanú alokované napr. singletony, cache, ktoré by pri dalokácii zbytočne zdržovali ukončenie aplikácie.
Memory leak nebýva jediným dôvodom prečo môže rásť množstvo alokovanej pamäte. Môže to byť aj v dôsledku iných vecí napr. cache, kotrá sa nikdy neuvoľňuje a pravidelne sa plní novými údajmi (pred týmto typom chyby neochráni ani jazyk s GC). Preto často namiesto použitia špecializovaných nástrojov ako valgrind radšej nechám svoju aplikáciu v slučke vykonávať nejakú akciu a sledujem, či mi nerastie množstvo alokovanej pamäte. Takto dokážem odhaliť akékoľvek úniky pamäte, aj tie, pred ktorými by ma nechránil GC.
Teraz k tomu Qt príkladu. Najskôr som si vytvoril kostru, ktorá vykoná určitý počet iterácií určitej akcie. Je to škaredý kód (napr. lambda je tam zbytočne), ale umožňuje nastaviť napríklad čakanie aby aplikácia mala čas na zobrazenie okna, prípadne aj s animáciou. Kostra vyzerá takto:
#include <QtWidgets/QApplication> #include <QtCore/QEventLoop> #include <QtCore/QTimer> int counter = 0; const int cycle_count = 1; const int cycle_duration = 1; void show_charts() { // Kód QEventLoop loop; QTimer::singleShot(cycle_duration, &loop, SLOT(quit())); loop.exec(); counter++; if (counter >= cycle_count) { qApp->quit(); } } int main(int argc, char *argv[]) { QApplication a(argc, argv); a.setQuitOnLastWindowClosed(false); QTimer timer; timer.setInterval(cycle_duration); timer.start(); QObject::connect(&timer, &QTimer::timeout, []() { show_charts(); }); return a.exec(); }
Na začiatok som skúsil 1 iteráciu zobrazenia okna [a.cpp]. Aplikáciu som spustil cez nástroj memusage:
memusage --png=a_1.png ./binarka
Vo výslednom grafe [a_1.png] je alokácia pamäte pre jedno okno.
Druhý pokus som urobil s 1 000 oknami. V grafe [a_2.png] je krásne vidieť, že maximálna alokovaná pamäť pri každej iterácii je konštantná.
To isté som zopakoval s kódom [b.cpp] (tento kód s grafmi) a pamäť opäť zostala konštantná [b.png].
Z nameraných údajov preto odhadujem, že hlásené straty pamäte súvisia len so singletonmi, cache a podobnými jednorázovými alokáciami, ktoré neboli uvoľnené pri ukončení.
Viem, že je to lamerský spôsob, ale všetky ostatné nástroje sa mi zdali skôr kontraproduktívne.
Dík. Asi to budu tedy dělat takhle.
Naštěstí se mi povedlo ty Qt Charts v programu trochu izolovat, takže můžu spouštět program i tak, aby se žádné grafy nekreslily, a pak mi dobře funguje i ten ASan/Valgrind, takže většinu kódu můžu otestovat pomocí nich. A na ověření kódu pro grafy použiji ten tvůj postup s memusage
.
Tiskni
Sdílej:
ISSN 1214-1267, (c) 1999-2007 Stickfish s.r.o.