Hi, so far I've used DepthAI in Python without problems.
Lately I have been trying to integrate DepthAI in a multi-threaded C++ application. Basically I want to be able to stream multiple OAK cameras after selecting them based on their MxId.

I am seeing some weird behavior which I think is caused by the fact that the application itself is multi-threaded, or at least I can't find other explanations for it, but I don't understand exactly what the problem is.

This is a short summary of relevant code:

#include <depthai/depthai.hpp>

oak_camera::oak_camera(const std::string mx_id)
        : mx_id(mx_id), thread(&oak_camera::stream, this)
{
}

oak_camera::~oak_camera()
{
        thread.join();
}

void oak_camera::stream()
{
    dai::Pipeline pipeline;

    // define some pipeline here...
    
    bool found;
    dai::DeviceInfo device_info;
    std::tie(found, device_info) = dai::Device::getDeviceByMxId(mx_id);
    dai::Device device{pipeline, device_info};

    // start showing frames here....
}    

I'll try to explain here the problems I had so far:

  1. the function (found, device_info) = dai::Device::getDeviceByMxId(mx_id) would return "found" as false even if the camera is connected. If I ignore "found" and initialize a device with the retrieved device_info it works anyway and the correct camera is started.

  2. If I use an "if" statement before initializing the device, something like:

    if (mx_id == something)
    {
        dai::Device device{pipeline, device_info};
    }
    else
    {
        dai::Device device{pipeline};
    }

    the "else" part doesn't work, throwing the error No available devices. If I remove the if statement and use either of the device initializations on their own it works.

I'm not sure what the problem is. The pipeline and initialization should be run in the same thread and at the moment I'm testing with only one camera so there isn't multiple threads running it.

Anyone could have any idea what the problem is?

    Hi Letty

    Are you working with USB or PoE device? Also which version depthai version do you use?

    Regarding number 1. - if its a PoE device, searching it by MX ID isn't yet available (but its searched by IP instead)

    Regarding number 2. - Does putting the same exact code and running same experiment in main thread work as intended? Does main thread wait for all oak_camera objects to destruct properly?
    The second case (else statement) should try connecting to any available device, so as long as a device is connected but not yet connected to by another Device object, it should work as inteded.
    If that doesn't seem to be the case, please provide a minimum reproducible example using the latest depthai release or develop and we'll try to reproduce.

    Regards,
    Martin

      themarpe

      Thank you for your reply.
      I'm working with USB cameras, so for point 1 I am searching for a camera through USB.
      I will do some experiments now by putting the code in the main thread and see how it behaves, I will get back when I have more information

      themarpe
      I forgot to mention that I'm using the latest depthai, 2.11.1

      11 days later

      themarpe

      Sorry for the delay in getting back on this.

      So I have it working now, but it seems that the problem wasn't the threading itself but something else that I have to determine.
      The code works when built in release mode but not in debug mode. In debug, this error is thrown by address sanitizer when trying to initialize a device

      [2021-11-23 11:17:12.817395] [0x00007fffdf427700] [error]   Reading from an OAK camera over USB failed: Cannot find any device with given deviceInfo
      ==14144==ERROR: AddressSanitizer failed to deallocate 0x800000 (8388608) bytes at address 0x7fffde426800
      ==14144==AddressSanitizer CHECK failed: ../../../../src/libsanitizer/sanitizer_common/sanitizer_posix.cc:60 "(("unable to unmap" && 0)) != (0)" (0x0, 0x0)
          #0 0x7ffff72bbbba  (/usr/lib/x86_64-linux-gnu/libasan.so.5+0x115bba)
          #1 0x7ffff72d9aca  (/usr/lib/x86_64-linux-gnu/libasan.so.5+0x133aca)
          #2 0x7ffff72cdfd5  (/usr/lib/x86_64-linux-gnu/libasan.so.5+0x127fd5)
          #3 0x7ffff72cf220  (/usr/lib/x86_64-linux-gnu/libasan.so.5+0x129220)
          #4 0x7ffff72c038c  (/usr/lib/x86_64-linux-gnu/libasan.so.5+0x11a38c)
          #5 0x7ffff2a42407 in __nptl_deallocate_tsd /build/glibc-S9d2JN/glibc-2.27/nptl/pthread_create.c:300
          #6 0x7ffff2a4381a in __nptl_deallocate_tsd ../sysdeps/nptl/futex-internal.h:200
          #7 0x7ffff2a4381a in start_thread /build/glibc-S9d2JN/glibc-2.27/nptl/pthread_create.c:473
          #8 0x7ffff1f7571e in __clone (/lib/x86_64-linux-gnu/libc.so.6+0x12171e)
      
      [1] + Done                       "/usr/bin/gdb" --interpreter=mi --tty=${DbgTerm} 0<"/tmp/Microsoft-MIEngine-In-sv5cjqg0.vzh" 1>"/tmp/Microsoft-MIEngine-Out-ouwv4g2m.quk"

      Not sure if this is also related, but when printing the device_info fields after device initialization obtained with either methods (getDeviceByMxId(mx_id) or assigned automatically)

      std::cout << device_info.desc.name << std::endl;
      std::cout << device_info.state << std::endl;
      std::cout << device_info.desc.protocol << std::endl;
      std::cout << device_info.desc.platform << std::endl;

      they will look like this

      14442C10A135C7D200-ma2480
      0
      0
      2480

      So it looks like the device state is always X_LINK_ANY_STATE either before or after assignment, which would explain why found is always false even if the device is connected.

        Letty
        Could you share a minimal example that reproduces this case?

          themarpe
          My application is big, I tried to build a smaller application to reproduce the problem but without success so far. I will share it as soon as I have it