Search code examples
pointersnvidiasycl

Using points in sycl class


Is it possible to copy a class containing pointers to its internal attribute using SYCL and offload it to the graphics card? Basically, I try to reference members to avoid unnecessary memory usage. I managed to have my example run on local memory but when offloading to a graphics card, problems occurred

PI CUDA ERROR:
        Value:           700
        Name:            CUDA_ERROR_ILLEGAL_ADDRESS
        Description:     an illegal memory access was encountered
        Function:        operator()
        Source Location: /root/intel-llvm-mirror/sycl/plugins/cuda/pi_cuda.cpp:2612

SYCL exception: Native API failed. Native API returns: -999 (Unknown PI error) -999 (Unknown PI error)

PI CUDA ERROR:
        Value:           700
        Name:            CUDA_ERROR_ILLEGAL_ADDRESS
        Description:     an illegal memory access was encountered
        Function:        wait
        Source Location: /root/intel-llvm-mirror/sycl/plugins/cuda/pi_cuda.cpp:653


PI CUDA ERROR:
        Value:           700
        Name:            CUDA_ERROR_ILLEGAL_ADDRESS
        Description:     an illegal memory access was encountered
        Function:        wait
        Source Location: /root/intel-llvm-mirror/sycl/plugins/cuda/pi_cuda.cpp:653


PI CUDA ERROR:
        Value:           700
        Name:            CUDA_ERROR_ILLEGAL_ADDRESS
        Description:     an illegal memory access was encountered
        Function:        get_next_transfer_stream
        Source Location: /root/intel-llvm-mirror/sycl/plugins/cuda/pi_cuda.cpp:513

I understand that this error is related to copying a memory pointer to another device, but given that I've provided an explicit class copy constructor, I would expect the copy to be a deep copy. Does SYCL fundamentally lack support for copying a pointer object to a graphics card? I've attached my code below

#include <array>
#include <iostream>
#include <sycl/sycl.hpp>
#include <vector>

using namespace sycl; // (optional) avoids the need for "sycl::" before SYCL names

typedef struct my1 {
    int value = 1;

    // Copy constructor that creates a new my1
    my1(const my1& other) : value(other.value) {}

    // Default constructor
    my1() = default;

    // Constructor with a value
    my1(int v) : value(v) {}
} my1;

typedef struct my2 {
    struct my1* a;
    struct my1* b;
} my2;

template <std::size_t N>
class MyClass {
public:
    MyClass(std::array<my1, N> a, std::array<my1, N> b) {
        for (std::size_t i = 0; i < N; i++) {
            _a[i].value = a[i].value;
            _b[i].value = b[i].value;
            _c[i].a = &_a[i];
            _c[i].b = &_b[i];
        }
    }

    float addValue(int i) const {
        if (i >= N || i >= N)
            // throw std::runtime_error("Index out of bounds");
            return 0;
        return (_c[i].a)->value + (_c[i].b)->value;
    }

    void modifyAValue(int i, float a) {
        if (i >= N)
            return;
        _a[i].value = a;
    }

    void modifyBValue(int i, float b) {
        if (i >= N)
            return;
        _b[i].value = b;
    }

    MyClass(const MyClass& other) {
        _c = std::array<my2, N>();
        for (std::size_t i = 0; i < N; i++) {
            _a[i].value = other._a[i].value;
            _b[i].value = other._b[i].value;
            _c[i].a = &_a[i];
            _c[i].b = &_b[i];
        }
    }

private:
    std::array<my1, N> _a;
    std::array<my1, N> _b;
    std::array<my2, N> _c;
};

template <std::size_t N>
struct sycl::is_device_copyable<MyClass<N>> : std::true_type {};

int main() {
    // Create a SYCL queue
    sycl::queue myQueue;

    // Create an instance of MyClass
    int N = 10;
    std::array<my1, 10> a{1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
    std::array<my1, 10> b{10, 9, 8, 7, 6, 5, 4, 3, 2, 1};
    MyClass<a.size()> myObject(a, b);

    std::vector<int> input = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9};
    std::vector<float> result(input.size());

    // Create a SYCL buffer to hold the result
    sycl::buffer<float, 1> resultBuffer(result.data(), sycl::range<1>(input.size()));
    sycl::buffer<int, 1> inputBuffer(input.data(), sycl::range<1>(input.size()));

    sycl::buffer<MyClass<10>> buffer(&myObject, 1);
    try {
        // Submit a SYCL kernel
        myQueue.submit([&](sycl::handler& cgh) {
            auto resultAcc = resultBuffer.template get_access<sycl::access::mode::write>(cgh);
            auto inputAcc = inputBuffer.template get_access<sycl::access::mode::read>(cgh);
            auto myObjectAcc = buffer.template get_access<sycl::access::mode::read>(cgh);

            cgh.parallel_for<class MyKernel>(
                sycl::range<1>(input.size()), [=](sycl::id<1> idx) {
                    // Call a member function on the instance of MyClass
                    resultAcc[idx] = myObjectAcc[0].addValue(inputAcc[idx]);
                });
        });

        // Wait for the kernel to complete and get the result
        myQueue.wait();
        myQueue.update_host(resultBuffer.get_access());
    } catch (const sycl::exception& e) {
        std::cerr << "SYCL exception: " << e.what() << std::endl;
        return 1;
    } catch (const std::runtime_error& e) {
        std::cerr << "Runtime error: " << e.what() << std::endl;
        return 1;
    }

    // Print the result (accumulated sum)
    float sum = 0.0f;
    for (float val : result) {
        std::cout << val << std::endl;
        sum += val;
    }
    std::cout << "Result: " << sum << std::endl;

    return 0;
}

In this example, I have a class that contains three arrays. Two of these arrays store information (my1), and the third one stores references to elements from the two information arrays (my2, which holds two pointers to my1 objects).

std::array<my1, N> _a;
std::array<my1, N> _b;
std::array<my2, N> _c;

As I understand it, errors occur because the kernel is attempting to access host memory. This is unexpected, as the constructor should have reassigned the pointer value after the copy operations.

MyClass(const MyClass& other) {
    _c = std::array<my2, N>();
    for (std::size_t i = 0; i < N; i++) 
    {
         _a[i].value = other._a[i].value;
         _b[i].value = other._b[i].value;
         _c[i].a = &_a[i];
         _c[i].b = &_b[i];
    }

I wouldn't mind using USM to allocate device memory, but I would prefer to use pointers or references to access the member variables inside the class. This approach helps avoid unnecessary memory allocation by referencing device memory directly (in short, I'd like to use pointers or references as member variable to access device memory)


Solution

  • After some research, I find it is impossible to emerge into SYCL without using usm or buffer below I have put the example code for this question.

    template <std::size_t N>
    class MyClass {
    public:
        MyClass(std::array<int, N> aValue, std::array<int, N> bValue,sycl::queue& queue): _queue(queue) {
    
            _a = sycl::malloc_shared<my1*>(N, _queue);
            _b = sycl::malloc_shared<my1*>(N, _queue);
            _c = sycl::malloc_shared<my2*>(N, _queue);
    
            for(int i = 0;i<N;i++)
            {
                _a[i] = sycl::malloc_shared<my1>( 1, _queue);
                _a[i]->value = aValue[i];
                _b[i] = sycl::malloc_shared<my1>( 1, _queue);
                _b[i]->value = bValue[i];
                _c[i] = sycl::malloc_shared<my2>( 1, _queue);
                _c[i]->a = _a[i];
                _c[i]->b = _b[i];          
            }
            std::cout<<"here"<<std::endl;
        }
    
        float addValue(int i) const {
            if (i >= N || i >= N)
                // throw std::runtime_error("Index out of bounds");
                return 0;
            return (_c[i]->a)->value + (_c[i]->b)->value;
        }
    
        void modifyAValue(int i, float a) {
            if (i >= N)
                return;
            _a[i]->value = a;
        }
    
        void modifyBValue(int i, float b) {
            if (i >= N)
                return;
            _b[i]->value = b;
        }
    
        MyClass(const MyClass& other) {
    
            for (int i = 0;i<N;i++)
            {
                _a[i]->value = other._a[i]->value;
                _b[i]->value = other._b[i]->value;
            }
    
        }
    
        ~MyClass() {
            for (int i = 0;i<N;i++)
            {
                sycl::free(_a[i], _queue);
                sycl::free(_b[i], _queue);
                sycl::free(_c[i], _queue);
            }
            sycl::free(_a, _queue);
            sycl::free(_b, _queue);
            sycl::free(_c, _queue);
    
    
        }
    
    private:
        my1** _a;
        my1** _b;
        my2** _c;
        sycl::queue& _queue;
    };