C ++ OpenCV AddressSanitizer: SEGV по неизвестному адресу 0x000000000040 - PullRequest
0 голосов
/ 24 мая 2018

Я не могу разобраться с этой ошибкой.Я работаю над проектом с использованием opencv, пытаясь использовать hog и svm, вдохновленные этим примером: https://docs.opencv.org/3.4.1/d5/d77/train_HOG_8cpp-example.html#_a33 (код действительно похож).Ошибка находится во второй строке этой функции на imgs [0].Я понял, что ошибка есть, потому что я попытался напечатать все изображение с cout << imgs[0];, и оно не печатает всю матрицу, но ошибка появляется в определенный момент.

void computeHOG(const vector<Mat> &imgs, vector<Mat> &gradients) {
  HOGDescriptor hog;
  hog.winSize = imgs[0].size() / 8 * 8;
  vector<float> descriptors;
  Mat gray;

  for (size_t i = 0; i < imgs.size(); ++i) {
    if (imgs[i].rows >= hog.winSize.height && imgs[i].cols >= hog.winSize.width) {
      Rect r = Rect((imgs[i].rows - hog.winSize.height) / 2,
                    (imgs[i].cols - hog.winSize.width) / 2,
                    hog.winSize.width,
                    hog.winSize.height);

      cvtColor(imgs[i](r), gray, COLOR_BGR2GRAY);
      hog.compute(gray, descriptors, Size(8, 8), Size(0, 0));
      gradients.push_back(Mat(descriptors).t());
    }
  }
}

Это мой Makefile

LDIR=lib
ODIR=build

CXX=g++
CXXFLAGS= -std=c++11 -Wall -Wextra -fsanitize=address -I$(LDIR)

LIBS=-lopencv_calib3d -lopencv_core -lopencv_features2d -lopencv_flann -lopencv_highgui -lopencv_imgcodecs -lopencv_imgproc -lopencv_ml -lopencv_objdetect -lopencv_photo -lopencv_shape -lopencv_stitching -lopencv_superres -lopencv_video -lopencv_videoio

_DEPS=io.hpp hog.hpp
DEPS=$(patsubst %,$(LDIR)/%,$(_DEPS))

_OBJ=main.o io.o hog.o
OBJ=$(patsubst %,$(ODIR)/%,$(_OBJ))

$(ODIR)/%.o: $(LDIR)/%.cpp
    $(CXX) -c -o $@ $< $(CXXFLAGS)

$(ODIR)/%.o: main.cpp $(DEPS)
    $(CXX) -c -o $@ $< $(CXXFLAGS)

main: $(OBJ)
    $(CXX) -o $@ $^ $(CXXFLAGS) $(LIBS)

clean:
    rm -f $(ODIR)/*.o

.PHONY: clean

Он компилируется, но когда я запускаю его, я получаю:

ASAN:SIGSEGV
=================================================================
==670==ERROR: AddressSanitizer: SEGV on unknown address 0x000000000040 (pc 0x00000040c917 bp 0x7ffe8906f990 sp 0x7ffe8906f980 T0)
    #0 0x40c916 in cv::MatSize::operator()() const (/[...]/opencv-project/main+0x40c916)
    #1 0x40ada0 in computeHOG(std::vector<cv::Mat, std::allocator<cv::Mat> > const&, std::vector<cv::Mat, std::allocator<cv::Mat> >&) (/[...]/opencv-project/main+0x40ada0)
    #2 0x4030df in main (/[...]/opencv-project/main+0x4030df)
    #3 0x7f511da2082f in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x2082f)
    #4 0x402b48 in _start (/[...]/opencv-project/main+0x402b48)

AddressSanitizer can not provide additional info. SUMMARY: AddressSanitizer: SEGV ??:0 cv::MatSize::operator()() const
==670==ABORTING

Если я удаляю -fsanitize=address из Makefile, ошибка возникает из-за компиляции, она слишком велика для вставкиздесь, но начинается с чего-то подобного

main.cpp:(.text+0x20): undefined reference to `__asan_option_detect_stack_use_after_return'
main.cpp:(.text+0x30): undefined reference to `__asan_stack_malloc_4'
build/main.o: In function `__static_initialization_and_destruction_0(int, int)':
main.cpp:(.text+0x970): undefined reference to `__asan_before_dynamic_init'
main.cpp:(.text+0x9d4): undefined reference to `__asan_report_load1'
main.cpp:(.text+0xa14): undefined reference to `__asan_report_store1'
main.cpp:(.text+0xa64): undefined reference to `__asan_report_load1'

Моя версия g ++ g++ (Ubuntu 5.4.0-6ubuntu1~16.04.9) 5.4.0 20160609 Не нашел никакого решения, ищущего это ...

1 Ответ

0 голосов
/ 24 мая 2018

Ваш код обращается к NULL указателю в некоторый момент при вызове Mat::operator ().Причины этого неясны из отправленного вами фрагмента, но должны быть очевидны, если вы запускаете свой код из отладчика.

Неопределенные ссылки при удалении -fsanitize=address предполагают, что какой-то код не был перекомпилирован после того, как вы изменили Makefile.Можете ли вы сначала сделать make clean?

В качестве примечания, вы должны также добавить -fsanitize=address к LDFLAGS (для связи в библиотеке времени выполнения).

...