日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

[转]OpenCL 教学(一)

發(fā)布時間:2023/12/18 编程问答 39 豆豆
生活随笔 收集整理的這篇文章主要介紹了 [转]OpenCL 教学(一) 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

Contents

  • OpenCL簡介
  • OpenCL的架構
  • OpenCL環(huán)境設定
  • 開始撰寫OpenCL程式
  • 建立Command Queue
  • 產(chǎn)生資料
  • 配置記憶體并復制資料
  • 編譯OpenCL kernel程式
  • 執(zhí)行OpenCL kernel
  • penCL 簡介
    OpenCL是由Khronos Group針對異質性計算裝置(heterogeneous device)進行平行化運算所設計的標準API以及程式語言。所謂的「異質性計算裝置」,是指在同一個電腦系統(tǒng)中,有兩種以上架構差異很大的計算裝置,例如一般的CPU以及顯示晶片,或是像CELL的PPE以及SPE。目前,最為常見的就是所謂的GPGPU應用,也就是利用一般的顯示晶片(即GPU)進行3D繪圖以外的計算工作。

    過去GPGPU的應用,有各種不同的使用方式。最早的GPGPU,多半是直接透過3D繪圖的API進行,例如OpenGL或D3D的HLSL(High Level Shading Language)。但是,這樣做有很多缺點,主要是即使想要進行的運算和3D繪圖無關,仍需要處理很多3D繪圖方面的動作(例如建立texture,安排render-to-texture動作等等)。這讓GPGPU變得十分復雜。后來開始有些嘗試把這些3D繪圖部份隱藏起來的想法,例如由Stanford大學設計的BrookGPU,可以透過不同的backend將Brook程式轉換成由CPU、Direct3D、或OpenGL來執(zhí)行。另外,也有各家顯示卡廠商自行開發(fā)的系統(tǒng),包括ATI針對其產(chǎn)品設計的Close to Metal(以及后來的AMD Stream),以及NVIDIA的CUDA。Microsoft也在DirectX 11中加入了特別為GPGPU設計的DirectCompute。

    由于各家廠商的GPGPU 方案都是互不相容的(例如AMD Stream 的程式無法在NVIDIA 的顯示晶片上執(zhí)行,而CUDA 的程式也不能在AMD 的顯示晶片上執(zhí)行),這對GPGPU 的發(fā)展是不利的,因為程式開發(fā)者必須為不同廠商的顯示晶片分別撰寫程式,或是選擇只支援某個顯示晶片廠商。由于顯示晶片的發(fā)展愈來愈彈性化,GPGPU 的應用范圍也增加,因此Apple 決定提出一個統(tǒng)一的GPGPU 方案。這個方案得到包括AMD、IBM、Intel、NVIDIA 等相關廠商的支持,并很快就交由Khronos Group 進行標準化。整個計畫只花了五個月的時間,并在2008 年十二月時正式公開。第一個正式支援OpenCL 的作業(yè)系統(tǒng)是Apple 的MacOS X 10.6 "Snow Leopard"。AMD 和NVIDIA 也隨后推出了在Windows 及Linux 上的OpenCL 實作。IBM 也推出了支援CELL 的OpenCL 實作。

    OpenCL 的主要設計目的,是要提供一個容易使用、且適用于各種不同裝置的平行化計算平臺。因此,它提供了兩種平行化的模式,包括task parallel 以及data parallel。目前GPGPU 的應用,主要是以data parallel 為主,這里也是以這個部份為主要重點。所謂的data parallel,指的是有大量的資料,都進行同樣的處理。這種形式的平行化,在很多工作上都可以見到。例如,影像處理的程式,經(jīng)常要對一個影像的每個pixel 進行同樣的動作(例如Gaussian blur)。因此,這類工作很適合data parallel 的模式。

    OpenCL 的架構

    OpenCL 包括一組API 和一個程式語言。基本上,程式透過OpenCL API 取得OpenCL 裝置(例如顯示晶片)的相關資料,并將要在裝置上執(zhí)行的程式(使用OpenCL 程式語言撰寫)編繹成適當?shù)母袷?#xff0c;在裝置上執(zhí)行。OpenCL API 也提供許多裝置控制方面的動作,例如在OpenCL 裝置上取得一塊記憶體、把資料從主記憶體復制到OpenCL 裝置上(或從OpenCL 裝置上復制到主記憶體中)、取得裝置動作的資訊(例如上一個程式執(zhí)行所花費的時間)等等。

    例如,我們先考慮一個簡單的工作:把一群數(shù)字相加。在一般的C 程式中,可能是如下:

    float a[DATA_SIZE];

    float b[DATA_SIZE];

    float result[DATA_SIZE];

    // ...

    for(int i = 0; i < DATA_SIZE; i++) {

    result[i] = a[i] + b[i];

    }在OpenCL 中,則大致的流程是:
  • 把OpenCL 裝置初始化。
  • 在OpenCL 裝置上配置三塊記憶體,以存放a、b、c 三個陣列的資料。
  • 把a 陣列和b 陣列的內(nèi)容,復制到OpenCL 裝置上。
  • 編譯要執(zhí)行的OpenCL 程式(稱為kernel)。
  • 執(zhí)行編譯好的kernel。
  • 把計算結果從OpenCL 裝置上,復制到result 陣列中。
  • 透過data parallel 的模式,這里的OpenCL 程式非常簡單,如下所示:

    __kernel void adder(__global const float* a, __global const float* b, __global float* result)

    {

    int idx = get_global_id(0);

    result[idx] = a[idx] + b[idx];

    }
    在一般的版本中,是透過一個回圈,執(zhí)行DATA_SIZE次數(shù)的加法動作。而在OpenCL中,則是建立DATA_SIZE個work item,每個work item都執(zhí)行上面所示的kernel。可以看到,OpenCL程式語言和一般的C語言非常類似。__kernel表示這個函式是在OpenCL裝置上執(zhí)行的。__global則表示這個指標是在global memory中(即OpenCL裝置上的主要記憶體)。而get_global_id(0)會傳回work item的編號,例如,如果有1024個work item,則編號會分別是0 ~ 1023(實際上編號可以是二維或三維,但在這里先只考慮一維的情形)。

    要如何讓上面這個簡單的OpenCL kernel 實際在OpenCL 裝置上執(zhí)行呢?這就需要透過OpenCL API 的幫助了。以下會一步一步說明使用OpenCL API 的方法。

    OpenCL 環(huán)境設定

    在使用OpenCL API 之前,不免要進行一些環(huán)境的設定。相關的動作可以參考下列的文章:

    • 在Windows 下使用OpenCL
    • 在Xcode 中使用OpenCL
    開始撰寫OpenCL 程式

    在使用OpenCL API之前,和絕大部份所有其它的API一樣,都需要include相關的header檔案。由于在MacOS X 10.6下OpenCL的header檔案命名方式和在其它作業(yè)系統(tǒng)下不同,因此,通常要使用一個#ifdef來進行區(qū)分。如下所示:

    #ifdef __APPLE__

    #include <OpenCL/opencl.h>

    #else

    #include <CL/cl.h>

    #endif

    這樣就可以在MacOS X 10.6 下,以及其它的作業(yè)系統(tǒng)下,都可以include 正確的OpenCL header 檔。

    接著,要先取得系統(tǒng)上所有的OpenCL platform。在MacOS X 10.6 下,目前只有一個由Apple 提供的OpenCL platform,但是在其它系統(tǒng)上,可能會有不同廠商提供的多個不同的OpenCL platform,因此需要先取得platform 的數(shù)目:

    cl_int err;

    cl_uint num;

    err = clGetPlatformIDs(0, 0, &num);

    if(err != CL_SUCCESS) {

    std::cerr << "Unable to get platforms\n";

    return 0;

    }

    大部份的OpenCL API 會傳回錯誤值。如果傳回值是CL_SUCCESS 則表示執(zhí)行成功,否則會傳回某個錯誤值,表示失敗的原因。

    接著,再取得platform 的ID,這在建立OpenCL context 時會用到:

    std::vector<cl_platform_id> platforms(num);

    err = clGetPlatformIDs(num, &platforms[0], &num);

    if(err != CL_SUCCESS) {

    std::cerr << "Unable to get platform ID\n";

    return 0;

    }

    在OpenCL 中,類似這樣的模式很常出現(xiàn):先呼叫第一次以取得數(shù)目,以便配置足夠的記憶體量。接著,再呼叫第二次,取得實際的資料。

    接下來,要建立一個OpenCL context。如下:

    cl_context_properties prop[] = { CL_CONTEXT_PLATFORM, reinterpret_cast<cl_context_properties>(platforms[0]), 0 };

    cl_context context = clCreateContextFromType(prop, CL_DEVICE_TYPE_DEFAULT, NULL, NULL, NULL);

    if(context == 0) {

    std::cerr << "Can't create OpenCL context\n";

    return 0;

    }
    clReleaseContext(context);
    return 0; 在上面的程式中,clCreateContextFromType是一個OpenCL的API,它可以從指定的裝置類別中,建立一個OpenCL context。第一個參數(shù)是指定context的property。在OpenCL中,是透過一個property的陣列,以「property種類」及「property內(nèi)容」成對出現(xiàn),并以0做為結束。例如,以上面的例子來說,要指定的property種類是CL_CONTEXT_PLATFORM,即要使用的platform ID,而property內(nèi)容則是由之前取得的platform ID中的第一個(即platforms[0])。由于property的內(nèi)容可能是不同的資料型態(tài),因此需要使用reinterpret_cast來進行強制轉型。

    第二個參數(shù)可以指定要使用的裝置類別。目前可以使用的類別包括:

    • CL_DEVICE_TYPE_CPU:使用CPU 裝置
    • CL_DEVICE_TYPE_GPU:使用顯示晶片裝置
    • CL_DEVICE_TYPE_ACCELERATOR:特定的OpenCL 加速裝置,例如CELL
    • CL_DEVICE_TYPE_DEFAULT:系統(tǒng)預設的OpenCL 裝置
    • CL_DEVICE_TYPE_ALL:所有系統(tǒng)中的OpenCL 裝置

    這里使用的是CL_DEVICE_TYPE_DEFAULT,也就是指定使用預設的裝置。另外,在這里,直接使用了之前取得的OpenCL platform ID中的第一個ID(實際的程式中,可能會需要讓使用者可以指定要使用哪一個platform)。

    如果建立OpenCL context失敗,會傳回0。因此,要進行檢查,并顯示錯誤訊息。如果建立成功的話,在使用完后,要記得將context釋放。這可以透過呼叫clReleaseContext來達成。

    這個程式基本上已經(jīng)可以編譯執(zhí)行了,但是當然它并沒有真的做什么事情。

    一個OpenCL context中可以包括一個或多個裝置,所以接下來的工作是要取得裝置的列表。要取得任何和OpenCL context相關的資料,可以使用clGetContextInfo函式。以下是取得裝置列表的方式:

    size_t cb;

    clGetContextInfo(context, CL_CONTEXT_DEVICES, 0, NULL, &cb);

    std::vector<cl_device_id> devices(cb / sizeof(cl_device_id));

    clGetContextInfo(context, CL_CONTEXT_DEVICES, cb, &devices[0], 0);

    CL_CONTEXT_DEVICES表示要取得裝置的列表。和前面取得platform ID的情形相同,clGetContextInfo被呼叫了兩次:第一次是要取得需要存放裝置列表所需的記憶體空間大小(也就是傳入&cb),然后第二次呼叫才真正取得所有裝置的列表。

    接下來,可能會想要確定倒底找到的OpenCL裝置是什么。所以,可以透過OpenCL API取得裝置的名稱,并將它印出來。取得和裝置相關的資料,是使用clGetDeviceInfo函式,和前面的clGetContextInfo函式相當類似。以下是取得裝置名稱的方式:

    clGetDeviceInfo(devices[0], CL_DEVICE_NAME, 0, NULL, &cb);

    std::string devname;

    devname.resize(cb);

    clGetDeviceInfo(devices[0], CL_DEVICE_NAME, cb, &devname[0], 0);

    std::cout << "Device: " << devname.c_str() << "\n";

    到目前為止,完整的程式應該如下所示:

    // OpenCL tutorial 1

    #include <iostream>

    #include <string>

    #include <vector>

    #ifdef __APPLE__

    #include <OpenCL/opencl.h>

    #else

    #include <CL/cl.h>

    #endif

    int main()

    {

    cl_int err;

    cl_uint num;

    err = clGetPlatformIDs(0, 0, &num);

    if(err != CL_SUCCESS) {

    std::cerr << "Unable to get platforms\n";

    return 0;

    }


    std::vector<cl_platform_id> platforms(num);

    err = clGetPlatformIDs(num, &platforms[0], &num);

    if(err != CL_SUCCESS) {

    std::cerr << "Unable to get platform ID\n";

    return 0;

    }


    cl_context_properties prop[] = { CL_CONTEXT_PLATFORM, reinterpret_cast<cl_context_properties>(platforms[0]), 0 };

    cl_context context = clCreateContextFromType( prop , CL_DEVICE_TYPE_DEFAULT, NULL, NULL, NULL); if(context == 0) { std::cerr << "Can't create OpenCL context\n"; return 0; } size_t cb; clGetContextInfo(context, CL_CONTEXT_DEVICES, 0, NULL, &cb); std::vector<cl_device_id> devices(cb / sizeof(cl_device_id)); clGetContextInfo(context, CL_CONTEXT_DEVICES, cb, &devices[0], 0); clGetDeviceInfo(devices[0], CL_DEVICE_NAME, 0, NULL, &cb); std::string devname; devname.resize(cb); clGetDeviceInfo(devices[0], CL_DEVICE_NAME, cb, &devname[0], 0); std::cout << "Device: " << devname.c_str() << "\n"; clReleaseContext(context); return 0;

    }

    執(zhí)行這個程式,如果建立OpenCL context 成功的話,應該會顯示出找到的OpenCL 裝置的名稱,例如

    Device: GeForce GTX 285

    建立Command Queue

    大部份OpenCL 的操作,都要透過command queue。Command queue 可以接收對一個OpenCL 裝置的各種操作,并按照順序執(zhí)行(OpenCL 也容許把一個command queue 指定成不照順序執(zhí)行,即out-of-order execution,但是這里先不討論這個使用方式)。所以,下一步是建立一個command queue:

    cl_command_queue queue = clCreateCommandQueue(context, devices[0], 0, 0);

    if(queue == 0) {

    std::cerr << "Can't create command queue\n";
    clReleaseContext(context);

    return 0;

    } 和context 一樣,在程式結束前,要把command queue 釋放,即:

    clReleaseCommandQueue(queue);

    上面的程式中,是把裝置列表中的第一個裝置(即devices[0])建立command queue。如果想要同時使用多個OpenCL裝置,則每個裝置都要有自己的command queue。
    產(chǎn)生資料

    由于這個程式的目的是要把一大堆數(shù)字進行相加,所以需要產(chǎn)生一些「測試資料」:

    const int DATA_SIZE = 1048576;

    std::vector<float> a(DATA_SIZE), b(DATA_SIZE), res(DATA_SIZE);

    for(int i = 0; i < DATA_SIZE; i++) {

    a[i] = std::rand();

    b[i] = std::rand();

    }
    配置記憶體并復制資料

    要使用OpenCL 裝置進行運??算時,通常會需要在OpenCL 裝置上配置記憶體,并把資料從主記憶體中復制到裝置上。有些OpenCL 裝置可以直接從主記憶體存取資料,但是速度通常會比較慢,因為OpenCL 裝置(例如顯示卡)通常會有專用的高速記憶體。以下的程式配置三塊記憶體:

    cl_mem cl_a = clCreateBuffer(context, CL_MEM_READ_ONLY | CL_MEM_COPY_HOST_PTR, sizeof(cl_float) * DATA_SIZE, &a[0], NULL);

    cl_mem cl_b = clCreateBuffer(context, CL_MEM_READ_ONLY | CL_MEM_COPY_HOST_PTR, sizeof(cl_float) * DATA_SIZE, &b[0], NULL);

    cl_mem cl_res = clCreateBuffer(context, CL_MEM_WRITE_ONLY, sizeof(cl_float) * DATA_SIZE, NULL, NULL);

    if(cl_a == 0 || cl_b == 0 || cl_res == 0) {

    std::cerr << "Can't create OpenCL buffer\n";

    clReleaseMemObject(cl_a);

    clReleaseMemObject(cl_b);

    clReleaseMemObject(cl_res);

    clReleaseCommandQueue(queue);

    clReleaseContext(context);

    return 0;

    }clCreateBuffer函式可以用來配置記憶體。它的第二個參數(shù)可以指定記憶體的使用方式,包括:
    • CL_MEM_READ_ONLY:表示OpenCL kernel 只會對這塊記憶體進行讀取的動作
    • CL_MEM_WRITE_ONLY:表示OpenCL kernel 只會對這塊記憶體進行寫入的動作
    • CL_MEM_READ_WRITE:表示OpenCL kernel 會對這塊記憶體進行讀取和寫入的動作
    • CL_MEM_USE_HOST_PTR:表示希望OpenCL 裝置直接使用指定的主記憶體位址。要注意的是,如果OpenCL 裝置無法直接存取主記憶體,它可能會將指定的主記憶體位址的資料復制到OpenCL 裝置上。
    • CL_MEM_ALLOC_HOST_PTR:表示希望配置的記憶體是在主記憶體中,而不是在OpenCL 裝置上。不能和CL_MEM_USE_HOST_PTR 同時使用。
    • CL_MEM_COPY_HOST_PTR:將指定的主記憶體位址的資料,復制到配置好的記憶體中。不能和CL_MEM_USE_HOST_PTR 同時使用。

    第三個參數(shù)是指定要配置的記憶體大小,以bytes為單位。在上面的程式中,指定的大小是sizeof(cl_float) * DATA_SIZE。

    第四個參數(shù)是指定主記憶體的位置。因為對cl_a和cl_b來說,在第二個參數(shù)中,指定了CL_MEM_COPY_HOST_PTR,因此要指定想要復制的資料的位址。cl_res則不需要指定。

    第五個參數(shù)是指定錯誤碼的傳回位址。在這里并沒有使用到。

    如果clCreateBuffer因為某些原因無法配置記憶體(例如OpenCL裝置上的記憶體不夠),則會傳回0。要釋放配置的記憶體,可以使用clReleaseMemObject函式。

    編譯OpenCL kernel 程式

    現(xiàn)在執(zhí)行OpenCL kernel 的準備工作已經(jīng)大致完成了。所以,現(xiàn)在剩下的工作,就是把OpenCL kernel 程式編釋并執(zhí)行。首先,先把前面提過的OpenCL kernel 程式,存放在一個文字檔中,命名為shader.cl:

    __kernel void adder(__global const float* a, __global const float* b, __global float* result)

    {

    int idx = get_global_id(0);

    result[idx] = a[idx] + b[idx];

    }

    要編譯這個kernel程式,首先要把檔案內(nèi)容讀進來,再使用clCreateProgramWithSource這個函式,然后再使用clBuildProgram編譯。如下所示:

    cl_program load_program(cl_context context, const char* filename)

    {

    std::ifstream in(filename, std::ios_bas??e::binary);

    if(!in.good()) {

    return 0;

    }
    // get file length
    in.seekg(0, std::ios_base::end);
    size_t length = in.tellg();
    in.seekg(0, std::ios_base::beg);
    // read program source
    std::vector<char> data(length + 1);
    in.read(&data[0], length);
    data[length] = 0;
    // create and build program?
    const char* source = &data[0];
    cl_program program = clCreateProgramWithSource(context, 1, &source, 0, 0);
    if(program == 0) { return 0;}
    if(clBuildProgram(program, 0, 0, 0, 0, 0) != CL_SUCCESS) { return 0;}
    return program;} 上面的程式,就是直接將檔案讀到記憶體中,再呼叫clCreateProgramWithSource建立一個program object。建立成功后,再呼叫clBuildProgram函式編譯程式。clBuildProgram函式可以指定很多參數(shù),不過在這里暫時沒有使用到。

    有了這個函式,在main 函式中,直接呼叫:

    cl_program program = load_program(context, "shader.cl");

    if(program == 0) {

    std::cerr << "Can't load or build program\n";

    clReleaseMemObject(cl_a);

    clReleaseMemObject(cl_b);

    clReleaseMemObject(cl_res);

    clReleaseCommandQueue(queue);

    clReleaseContext(context);

    return 0;

    }

    同樣的,在程式結束前,要記得將program object 釋放:

    clReleaseProgram(program);

    一個OpenCL kernel 程式里面可以有很多個函式。因此,還要取得程式中函式的進入點:

    cl_kernel adder = clCreateKernel(program, "adder", 0);

    if(adder == 0) {

    std::cerr << "Can't load kernel\n";

    clReleaseProgram(program);

    clReleaseMemObject(cl_a);

    clReleaseMemObject(cl_b);

    clReleaseMemObject(cl_res);

    clReleaseCommandQueue(queue);

    clReleaseContext(context);

    return 0;

    }和program object 一樣,取得的kernel object 也需要在程式結束前釋放:

    clReleaseKernel(adder);

    執(zhí)行OpenCL kernel

    弄了這么多,總算可以執(zhí)行OpenCL kernel程式了。要執(zhí)行kernel程式,只需要先設定好函式的參數(shù)。adder函式有三個參數(shù)要設定:

    clSetKernelArg(adder, 0, sizeof(cl_mem), &cl_a);

    clSetKernelArg(adder, 1, sizeof(cl_mem), &cl_b);

    clSetKernelArg(adder, 2, sizeof(cl_mem), &cl_res);

    設定參數(shù)是使用clSetKernelArg函式。它的參數(shù)很簡單:第一個參數(shù)是要設定的kernel object,第二個是參數(shù)的編號(從0開始),第三個參數(shù)是要設定的參數(shù)的大小,第四個參數(shù)則是實際上要設定的參數(shù)內(nèi)部。以這里的adder函式來說,三個參數(shù)都是指向memory object的指標。

    設定好參數(shù)后,就可以開始執(zhí)行了。如下:

    size_t work_size = DATA_SIZE;

    err = clEnqueueNDRangeKernel(queue, adder, 1, 0, &work_size, 0, 0, 0, 0);

    clEnqueueNDRangeKernel會把執(zhí)行一個kernel的動作加到command queue里面。第三個參數(shù)(1)是指定work item數(shù)目的維度,在這里就是一維。第五個參數(shù)是指定work item的總數(shù)目,也就是DATA_SIZE。后面的參數(shù)現(xiàn)在暫時先不用管。如果成功加入的話,會傳回CL_SUCCESS。否則會傳回錯誤值。

    在執(zhí)行kernel 被加到command queue 之后,就可能會開始執(zhí)行(如果command queue 現(xiàn)在沒有別的工作的話)。但是clEnqueueNDRangeKernel 是非同步的,也就是說,它并不會等待OpenCL 裝置執(zhí)行完畢才傳回。這樣可以讓CPU 在OpenCL 裝置在進行運算的同時,進行其它的動作。

    由于執(zhí)行的結果是在OpenCL 裝置的記憶體中,所以要取得結果,需要把它的內(nèi)容復制到CPU 能存取的主記憶體中。這可以透過下面的程式完成:

    if(err == CL_SUCCESS) {

    err = clEnqueueReadBuffer(queue, cl_res, CL_TRUE, 0, sizeof(float) * DATA_SIZE, &res[0], 0, 0, 0);

    }clEnqueueReadBuffer函式會把「將記憶體資料從OpenCL裝置復制到主記憶體」的動作加到command queue中。第三個參數(shù)表示是否要等待復制的動作完成才傳回,CL_TRUE表示要等待。第五個參數(shù)是要復制的資料大小,第六個參數(shù)則是目標的位址。

    由于這里指定要等待復制動作完成,所以當函式傳回時,資料已經(jīng)完全復制完成了。最后是進行驗證,確定資料正確:

    if(err == CL_SUCCESS) {

    bool correct = true;

    for(int i = 0; i < DATA_SIZE; i++) {

    if(a[i] + b[i] != res[i]) {

    correct = false;

    break;

    }}
    if(correct) { std::cout << "Data is correct\n"; }
    else {
    std::cout << "Data is incorrect\n";
    }}
    else { std::cerr << "Can't run kernel or read back data\n";} 到這里,整個程式就算是完成了。編譯后執(zhí)行,如果順利的話,應該會印出

    Data is correct

    的訊息。

    以下是整個程式的全貌:

    // OpenCL tutorial 1

    #include <iostream>

    #include <fstream>

    #include <string>

    #include <vector>

    #include <cstdlib>

    #ifdef __APPLE__

    #include <OpenCL/opencl.h>

    #else

    #include <CL/cl.h>

    #endif

    cl_program load_program(cl_context context, const char* filename)

    {

    std::ifstream in(filename, std::ios_bas??e::binary); if(!in.good()) { return 0; } // get file length in.seekg(0, std::ios_bas??e::end); size_t len??gth = in.tellg(); in.seekg(0, std::ios_bas??e::beg); // read program source std::vector<char> data(length + 1); in.read(&data[0], length); data[length] = 0; // create and build program const char* source = &data[0]; cl_program program = clCreateProgramWithSource(context, 1, &source, 0, 0); if(program == 0) { return 0; } if(clBuildProgram(program, 0, 0, 0, 0, 0) != CL_SUCCESS) { return 0; } return program;

    }

    int main()

    {

    cl_int err;

    cl_uint num;

    err = clGetPlatformIDs(0, 0, &num);

    if(err != CL_SUCCESS) {

    std::cerr << "Unable to get platforms\n";

    return 0;

    }


    std::vector<cl_platform_id> platforms(num);

    err = clGetPlatformIDs(num, &platforms[0], &num);

    if(err != CL_SUCCESS) {

    std::cerr << "Unable to get platform ID\n";

    return 0;

    }


    cl_context_properties prop[] = { CL_CONTEXT_PLATFORM, reinterpret_cast<cl_context_properties>(platforms[0]), 0 };

    cl_context context = clCreateContextFromType(prop, CL_DEVICE_TYPE_DEFAULT, NULL, NULL, NULL); if(context == 0) { std::cerr << "Can't create OpenCL context\n"; return 0; } size_t cb; clGetContextInfo(context, CL_CONTEXT_DEVICES, 0, NULL, &cb); std::vector<cl_device_id> devices(cb / sizeof(cl_device_id)); clGetContextInfo(context, CL_CONTEXT_DEVICES, cb, &devices[0], 0); clGetDeviceInfo(devices[0], CL_DEVICE_NAME, 0, NULL, &cb); std::string devname; devname.resize(cb); clGetDeviceInfo(devices[0], CL_DEVICE_NAME, cb, &devname[0], 0); std::cout << "Device: " << devname.c_str() << "\n"; cl_command_queue queue = clCreateCommandQueue(context, devices[0], 0, 0); if(queue == 0) { std::cerr << "Can't create command queue\n"; clReleaseContext(context); return 0; } const int DATA_SIZE = 1048576; std::vector<float> a(DATA_SIZE), b(DATA_SIZE), res(DATA_SIZE); for(int i = 0; i < DATA_SIZE; i++) { a[i] = std::rand(); b[i] = std::rand(); } cl_mem cl_a = clCreateBuffer(context, CL_MEM_READ_ONLY | CL_MEM_COPY_HOST_PTR, sizeof(cl_float) * DATA_SIZE, &a[0], NULL); cl_mem cl_b = clCreateBuffer(context, CL_MEM_READ_ONLY | CL_MEM_COPY_HOST_PTR, sizeof(cl_float) * DATA_SIZE, &b[0], NULL); cl_mem cl_res = clCreateBuffer(context, CL_MEM_WRITE_ONLY, sizeof(cl_float) * DATA_SIZE, NULL, NULL); if(cl_a == 0 || cl_b == 0 || cl_res == 0) { std::cerr << "Can't create OpenCL buffer\n"; clReleaseMemObject(cl_a); clReleaseMemObject(cl_b); clReleaseMemObject(cl_res); clReleaseCommandQueue(queue); clReleaseContext(context); return 0; } cl_program program = load_program(context, "shader.cl"); if(program == 0) { std::cerr << "Can't load or build program\n"; clReleaseMemObject(cl_a); clReleaseMemObject(cl_b); clReleaseMemObject(cl_res); clReleaseCommandQueue(queue); clReleaseContext(context); return 0; } cl_kernel adder = clCreateKernel(program, "adder", 0); if(adder == 0) { std::cerr << "Can't load kernel\n"; clReleaseProgram(program); clReleaseMemObject(cl_a); clReleaseMemObject(cl_b); clReleaseMemObject(cl_res); clReleaseCommandQueue(queue); clReleaseContext(context); return 0; } clSetKernelArg(adder, 0, sizeof(cl_mem), &cl_a); clSetKernelArg(adder, 1, sizeof(cl_mem), &cl_b); clSetKernelArg(adder, 2, sizeof(cl_mem), &cl_res); size_t work_size = DATA_SIZE; err = clEnqueueNDRangeKernel(queue, adder, 1, 0, &work_size, 0, 0, 0, 0); if(err == CL_SUCCESS) { err = clEnqueueReadBuffer(queue, cl_res, CL_TRUE, 0, sizeof(float) * DATA_SIZE, &res[0], 0, 0, 0); } if(err == CL_SUCCESS) { bool correct = true; for(int i = 0; i < DATA_SIZE; i++) { if(a[i] + b[i] != res[i]) { correct = false; break; } } if(correct) { std::cout << "Data is correct\n"; } else { std::cout << "Data is incorrect\n"; } } else { std::cerr << "Can't run kernel or read back data\n"; } clReleaseKernel(adder); clReleaseProgram(program); clReleaseMemObject(cl_a); clReleaseMemObject(cl_b); clReleaseMemObject(cl_res); clReleaseCommandQueue(queue); clReleaseContext(context); return 0;

    }

    在附件中可以下載包括Xcode project 以及Visual Studio 2008 project 檔的原始碼。

    Attachments ( 1 )
    • cltut_1.zip - on Feb 3, 2010 8:54 AM by Chen Ping-Che (version 2 / earlier versions ) 7k Download

    登錄 最近的站點活動 服務條款 舉報不良信息 打印頁面 由Google協(xié)作平臺強力驅動

    轉載于:https://www.cnblogs.com/wangshide/articles/2218354.html

    總結

    以上是生活随笔為你收集整理的[转]OpenCL 教学(一)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

    日韩1级片 | 在线小视频 | 五月婷影院 | 久久精品看| 欧美一二三区在线播放 | 中文av在线免费观看 | 天天干天天操天天拍 | 黄色免费网站大全 | 黄色成人在线网站 | 亚洲精品国产成人 | 日本黄色免费看 | 国产精品成人一区二区 | 国产护士hd高朝护士1 | 又黄又爽又刺激 | 国产精品麻豆一区二区三区 | 国产一区二区不卡视频 | 97在线视频观看 | 久久视了 | 国产大片免费久久 | 手机av观看 | 色婷婷av国产精品 | 在线导航av| 69av久久 | 国产精品 日韩精品 | 99视频| 2020天天干夜夜爽 | 国产精品视频观看 | 国产精品99久久久久久宅男 | 国产美腿白丝袜足在线av | 亚洲国产免费 | 18av在线视频 | 国产99久久精品一区二区永久免费 | 国产精品国产自产拍高清av | av电影久久 | 91九色在线观看视频 | 五月天电影免费在线观看一区 | 日韩精品专区在线影院重磅 | 免费毛片aaaaaa | 天天操狠狠操夜夜操 | 日日干影院 | 精品极品在线 | www.亚洲视频 | 亚洲国产精品女人久久久 | 国产精品色视频 | 久久理伦片 | 日日夜夜天天久久 | 日韩精品久久久久久久电影99爱 | 日韩黄色大片在线观看 | 香蕉精品视频在线观看 | 日本精品一区二区三区在线观看 | 在线视频精品 | 免费高清男女打扑克视频 | 美女精品网站 | 蜜臀av一区 | 国产香蕉97碰碰碰视频在线观看 | 日本一区二区三区免费看 | 91豆花在线 | 中文字幕亚洲精品在线观看 | 97福利| 成人毛片在线观看视频 | 日本一区二区高清不卡 | 日本少妇视频 | 国产欧美在线一区二区三区 | 精品99在线视频 | 黄色大全免费网站 | 欧美日韩一区二区久久 | 国产免码va在线观看免费 | 国产精品成人免费 | 一级电影免费在线观看 | 91精品国产91 | 27xxoo无遮挡动态视频 | 伊人久久在线观看 | 天天综合网国产 | 日韩极品视频在线观看 | 91麻豆精品一区二区三区 | 国产免费精彩视频 | 久久中文视频 | 日韩精品大片 | 国产成人av综合色 | 久草在线视频看看 | 91综合色 | 国产精品ⅴa有声小说 | 中文字幕视频播放 | 国产精品成人av久久 | 91亚洲国产成人 | 一区二区中文字幕在线观看 | 天天弄天天干 | 日韩黄色一级电影 | 丁香六月婷婷开心婷婷网 | 日韩欧美精选 | 国产一级视频在线观看 | 欧美一区中文字幕 | 日批视频 | 国产视频不卡一区 | 一区二区激情 | 久久久99精品免费观看乱色 | 久久久精品免费看 | 国产高清av在线播放 | 超薄丝袜一二三区 | 免费在线观看成年人视频 | 亚洲色图27p | 久久刺激视频 | 成人影片在线免费观看 | 亚洲精品资源在线观看 | 91系列在线观看 | 在线观看午夜 | 亚洲做受高潮欧美裸体 | 91在线成人 | ,久久福利影视 | 国产97在线观看 | 中文在线| 一区二区久久久久 | 中文在线字幕免 | 婷婷激情五月 | 亚洲动漫在线观看 | 日韩毛片在线免费观看 | 国产精品乱码一区二三区 | 香蕉视频在线视频 | 欧美日韩高清 | 国产一区欧美二区 | 精品99999| 久久久综合精品 | 在线看黄色的网站 | 国产成人精品av | 丁香婷婷激情啪啪 | 国产亚洲精品久久久久久网站 | 亚洲精品综合一二三区在线观看 | 黄污视频网站大全 | 久久久综合电影 | 日韩aⅴ视频| 中文字幕在线影院 | 国产高清视频免费最新在线 | 999成人 | 国产99久久久国产精品成人免费 | 一级一片免费视频 | 深夜免费福利视频 | 天天色天天射综合网 | 中文字幕在线看人 | 日韩久久精品一区二区三区下载 | 九九热在线精品视频 | 日韩在线视频免费观看 | 久久久久在线 | 成人免费在线电影 | 日韩av中文字幕在线免费观看 | 天天色天天干天天色 | 三级黄色片子 | 国产资源在线视频 | 中文字幕字幕中文 | 免费视频 你懂的 | 午夜精品成人一区二区三区 | 久久久国产精品久久久 | 最新国产在线视频 | 天天干夜夜擦 | 五月激情丁香图片 | 93久久精品日日躁夜夜躁欧美 | 国产精品视频免费在线观看 | 97超碰国产精品 | 狠狠的干狠狠的操 | www.超碰97.com| 射射射综合网 | 欧美日韩视频在线播放 | 色姑娘综合网 | 天天插狠狠插 | 久日精品 | 国产第一页在线播放 | 亚洲精品影视在线观看 | wwwwww国产| 激情av网址 | 激情网色| 91色视频| 伊人小视频 | 久久久91精品国产一区二区精品 | 国产性天天综合网 | 亚洲美女精品区人人人人 | 久久久一本精品99久久精品 | 天堂av在线网站 | 99re8这里有精品热视频免费 | 国内精品免费 | 久久人人爽人人 | 激情综合五月网 | 免费黄色在线网站 | 一级片黄色片网站 | 成人毛片在线观看 | 久久激情五月激情 | 日韩久久一区二区 | 久久精品99| 在线视频你懂得 | 日本性xxx | 另类五月激情 | 国产精品区一区 | 久久综合九色综合欧美就去吻 | 久久久久久久久久久久av | 亚洲精品免费视频 | 黄色网www | 国产一区二区三区免费在线 | 日本精品一区二区在线观看 | 日韩免费网站 | 人人澡人人模 | 精品中文字幕在线观看 | 天天插日日射 | 日韩精品免费在线 | 欧美精品一区二区在线播放 | 伊人中文网 | 婷婷丁香社区 | 日韩三区在线观看 | 国产成人专区 | 国内精品免费久久影院 | 天天爱综合 | 91精品无人成人www | 久久国产一区二区 | 91欧美国产 | 欧美日韩破处 | 欧美日韩国产欧美 | 日韩欧美高清不卡 | 黄色成人小视频 | 98久久 | 国内久久精品视频 | 色香蕉视频| 18av在线视频 | 免费日韩一级片 | 波多野结衣久久资源 | 狠狠的操狠狠的干 | 麻花天美星空视频 | 日韩国产精品一区 | 日韩成人黄色 | 国产色视频123区 | 亚洲综合在线视频 | 久久久久免费看 | 青草视频在线 | 在线看片视频 | 91成人在线网站 | 国产一区二区在线观看免费 | 99久久久久久久 | 日韩视频在线观看视频 | 日本三级全黄少妇三2023 | 欧美日韩国产一区二区三区在线观看 | 成人免费视频播放 | 欧美精品亚州精品 | 国产精品久久毛片 | 在线精品亚洲一区二区 | 亚洲成人黄色 | 精品人人人人 | 色橹橹欧美在线观看视频高清 | 丁香婷婷激情网 | 国内精品久久久久久久久久久久 | 天天色天天操天天爽 | 久久精品99久久久久久 | 婷婷丁香久久五月婷婷 | 精品在线不卡 | 午夜少妇一区二区三区 | 国产一级二级在线观看 | 国产成人一区二区三区 | 四虎成人精品永久免费av九九 | 亚洲狠狠婷婷综合久久久 | 国产精品久久一 | 狠狠干婷婷色 | 西西www4444大胆在线 | 国产精品国产亚洲精品看不卡15 | 国产精品免费久久久久久久久久中文 | www.91av在线 | 91麻豆精品国产91久久久更新时间 | 中文字幕 欧美性 | 国产一区自拍视频 | 日韩天堂在线观看 | 国产麻豆精品久久一二三 | 在线亚洲激情 | 伊人电影天堂 | 在线国产精品视频 | 91色在线观看视频 | 久草在线资源免费 | 色婷婷狠狠五月综合天色拍 | 中日韩男男gay无套 日韩精品一区二区三区高清免费 | 国产美女永久免费 | 992tv人人草 黄色国产区 | 精品99免费| 99久久精品国产欧美主题曲 | 天天爽夜夜爽精品视频婷婷 | 成人久久久久久久久久 | 在线观看久 | 视频一区二区在线 | 欧美日韩高清免费 | www.日本色 | 国产精品欧美久久久久无广告 | 成人免费在线视频观看 | 亚洲视频精品 | 黄色大片av| 欧美性生活免费 | av 在线观看 | 精品视频免费播放 | 久久综合免费 | 久久久久夜色 | 91中文字幕在线观看 | 亚洲电影成人 | 欧美激情奇米色 | 成人av电影网址 | 国产精品久久一区二区三区不卡 | 亚洲伦理一区 | 欧美日韩亚洲精品在线 | 中文字幕日韩一区二区三区不卡 | 久久免费a | 精品在线免费观看 | 97视频在线免费观看 | 亚洲一区日韩在线 | 成人av影视| www夜夜 | 69视频网站 | 日韩视频免费 | 久久99国产精品视频 | 黄色激情网址 | 久久久久久久久久久高潮一区二区 | 欧美日韩免费视频 | 日韩一区二区三区观看 | 日韩免费网址 | 精品一区二区免费视频 | 国产精品18p| 欧美在线视频日韩 | 国产专区视频 | 国产婷婷vvvv激情久 | 久久久久女人精品毛片九一 | 久久久久草 | 超碰在线观看av.com | 日本精油按摩3 | 国产老太婆免费交性大片 | 久久黄色a级片 | 在线观看福利网站 | 黄色免费av | 伊人天堂久久 | 操操日日 | 亚洲精品综合在线观看 | 久久玖| www.com久久| 中文字幕在线播出 | 欧美一区二区三区在线观看 | 国产精品一区二区三区久久久 | a黄色一级片 | 久草新在线 | 性日韩欧美在线视频 | 久久免费精品国产 | 一级片观看 | 91精品免费视频 | 91传媒91久久久 | 狠狠色噜噜狠狠狠狠2022 | www.国产精品| 国产视频在线观看一区 | 国产精品一区二区久久 | 亚洲欧美日本国产 | 综合久久精品 | 国产精品久久人 | 免费av影视 | 91精品国产99久久久久久红楼 | 欧美激情精品久久久 | 色综合久久久久综合 | 亚洲高清视频在线观看 | 丁香激情网 | 亚洲国产美女精品久久久久∴ | 色婷婷狠狠五月综合天色拍 | 99精品热| 亚洲播播| 91传媒免费观看 | 有码视频在线观看 | 欧美综合在线视频 | 伊人av综合| 黄色aa久久 | 亚洲狠狠丁香婷婷综合久久久 | 99久久精品免费看国产免费软件 | 久久久国产精品视频 | 综合网伊人 | 二区三区在线 | 自拍超碰在线 | 成人免费视频网站 | 欧美色综合久久 | 久久国产精品免费看 | 久99久精品视频免费观看 | 天天曰夜夜爽 | 日韩黄色软件 | 国产中文字幕视频在线 | www.com黄色| 97成人资源 | 99久久精品国产亚洲 | 蜜臀av性久久久久av蜜臀三区 | 色综合久久久久综合体 | 最新黄色av网址 | 成人四虎影院 | 国产精品一区一区三区 | 国产精品mv在线观看 | www在线观看视频 | 国产精品色视频 | 成人av一二三区 | 中文字幕在线播放一区 | 国产在线播放一区二区三区 | 999男人的天堂 | 国产一区91 | 亚洲精品乱码久久久久v最新版 | 亚洲午夜精品久久久久久久久 | 在线小视频你懂的 | 精品一区 精品二区 | 久久怡红院 | 欧美老人xxxx18 | av亚洲产国偷v产偷v自拍小说 | 天天摸日日摸人人看 | 国产精品久久久久久久久久 | 美女性爽视频国产免费app | 2019中文字幕第一页 | 婷婷丁香狠狠爱 | 一区二区三区中文字幕在线 | 麻豆传媒电影在线观看 | 久久综合九色九九 | 国产精品国产三级国产不产一地 | 亚色视频在线观看 | 黄色三级免费片 | 精品你懂的 | 日韩精品久久久久 | 欧美大片在线观看一区 | 日韩免费电影在线观看 | 国产精品自拍在线 | 五月婷婷伊人网 | 久草在线视频看看 | 国产中文字幕在线 | 免费看的黄色录像 | 天天色播 | 久久激情五月丁香伊人 | 色综合久久久网 | 婷婷激情综合五月天 | a视频免费| 亚洲成a人片在线观看中文 中文字幕在线视频第一页 狠狠色丁香婷婷综合 | 日产乱码一二三区别免费 | 中文字幕一区二区在线观看 | 五月婷婷天堂 | 成人免费看电影 | 国产精品嫩草影视久久久 | 免费男女羞羞的视频网站中文字幕 | 精品中文字幕在线播放 | 欧美韩日精品 | 夜夜操网| 久久久国产一区二区 | 国产成人久久av免费高清密臂 | 91精品1区2区 | 免费看污片 | 久久a国产 | 综合网av | 国产日产精品一区二区三区四区的观看方式 | 天天射天天操天天干 | 国产精品一区免费在线观看 | 高清不卡毛片 | 国产精品99久久久久久宅男 | 国内精品久久久久影院男同志 | 手机看片| 国产三级香港三韩国三级 | 久久精品直播 | av高清网站在线观看 | 91资源在线免费观看 | 天天天天天天干 | 成人av一区二区三区 | 国产精品 国产精品 | 色妞色视频一区二区三区四区 | 美女网站视频色 | av中文字幕网址 | 开心丁香婷婷深爱五月 | 最新国产在线 | 日韩一级黄色大片 | 久久av在线 | 欧美吞精 | 国产看片免费 | 久久久www成人免费精品张筱雨 | 亚洲国产成人精品在线观看 | 国产高清在线 | 亚洲一级在线观看 | 成人免费网站在线观看 | 在线观看免费av网站 | 久久电影网站中文字幕 | 911久久| 国内免费久久久久久久久久久 | 精品一区二区三区香蕉蜜桃 | 国产一区二区三区免费观看视频 | 尤物一区二区三区 | 97超碰在 | 久久国产精彩视频 | 激情文学丁香 | 久久精品看片 | 免费高清男女打扑克视频 | 日韩三区在线观看 | 久久黄色成人 | 日韩中文字幕在线 | www.久草.com| 中文字幕在线观看三区 | h网站免费在线观看 | 日韩性xxx| 午夜av免费| 日韩a级免费视频 | 中文字幕av免费在线观看 | 久久精品a | 国内久久精品 | 久久婷婷开心 | 99久久99久久精品国产片 | 日韩精品视频一二三 | 天天综合久久综合 | 91免费观看 | 手机在线小视频 | 亚洲国产中文字幕在线 | 丁香花中文在线免费观看 | 在线观看视频色 | 午夜精品导航 | 91精品中文字幕 | 超碰人人国产 | 免费av网站观看 | 深爱开心激情网 | 99tvdz@gmail.com| 国产无套精品久久久久久 | 免费看片网址 | 中文字幕av在线播放 | 亚洲精品国产精品国自产在线 | 日日干美女 | 久久精品视频免费播放 | 日韩久久久久久久久久久久 | 国产视频一区在线 | 丁香九月婷婷综合 | 欧美一级特黄高清视频 | 美女啪啪图片 | 天天射天 | 国产一区二区三区高清播放 | 国产成人精品久 | 一区二区三区三区在线 | 97国产在线视频 | 在线视频91 | 玖玖爱国产在线 | 国产免费高清 | 69av在线播放 | 成人四虎 | www.色的 | 五月婷香蕉久色在线看 | 特级毛片在线观看 | 韩国三级在线一区 | 91久久奴性调教 | 久久中文字幕视频 | 97人人精品| 免费看黄电影 | 欧美日韩在线播放 | 一区二区三区在线免费观看 | 久久久一本精品99久久精品 | 精品一区二区免费视频 | 中文字幕在线观看第一页 | 中文在线资源 | 国产美女久久久 | 国产精品久久久久久久电影 | av一级在线| 99精品免费在线观看 | 欧美国产日韩一区二区三区 | 国产午夜精品一区二区三区在线观看 | 国产福利资源 | 久久夜色精品国产欧美乱 | 国产免费av一区二区三区 | 一级黄色电影网站 | 成年人免费电影在线观看 | 亚洲精品乱码久久久久久蜜桃91 | 免费在线看v| 欧美色图狠狠干 | www.av在线.com | 综合色站| 精品福利片 | 天堂av在线网站 | 欧美一区二区三区在线播放 | 日日日日| 国产色在线视频 | 麻豆国产视频下载 | 久久久久国产一区二区三区四区 | 日韩在线精品一区 | 狠狠插狠狠操 | 国产色婷婷在线 | 欧美美女视频在线观看 | 色多多污污在线观看 | 九月婷婷综合网 | 中文字幕在线观看视频一区二区三区 | 色中色亚洲 | 欧美性大胆 | 99爱在线观看 | 狠狠操夜夜 | 伊人影院av | 亚洲一区二区三区四区在线视频 | www好男人 | 在线精品观看国产 | 成人a毛片| 最新av在线免费观看 | 日日夜夜精品免费 | 欧美日韩国产二区 | 99久久婷婷国产综合亚洲 | 成人91免费视频 | 亚洲免费在线观看视频 | 久久精品免费播放 | 欧美日韩另类在线观看 | 免费黄色av | 一区二区三区高清在线 | 久久99免费视频 | 日韩免费一区二区三区 | 精品视频不卡 | 99精品视频一区二区 | 丁香五月亚洲综合在线 | 91在线porny国产在线看 | 亚a在线 | 黄污网| 福利一区二区三区四区 | 国产麻豆精品95视频 | 色综合久久中文字幕综合网 | 国产视频1区2区3区 久久夜视频 | 超碰人人在线观看 | 青青河边草观看完整版高清 | 在线韩国电影免费观影完整版 | 欧美大片大全 | 欧美国产高清 | 懂色av一区二区在线播放 | 日韩视频一区二区在线 | 日本黄区免费视频观看 | 国产精品初高中精品久久 | 精品国产久| 69精品| 中文字幕精品一区久久久久 | 99在线观看视频网站 | 色婷婷骚婷婷 | 五月婷色| 日韩免费在线观看视频 | 国产一级免费在线 | 亚洲精品久久久蜜臀下载官网 | 在线国产精品视频 | 天天人人 | 久久午夜精品 | 高清免费在线视频 | 久久亚洲欧美 | 国产在线精品福利 | 欧美午夜久久 | 久久综合色婷婷 | 日韩欧在线 | 夜色资源网 | 免费网站观看www在线观看 | 色中色资源站 | 亚洲精品视频网址 | 久久99精品久久久久久秒播蜜臀 | 91综合色 | 国产小视频在线播放 | 欧美日韩国产免费视频 | 欧美精品黑人性xxxx | 国产精品av免费在线观看 | 免费观看一级视频 | 丁香av | 国产精品久久久久永久免费看 | 精品在线观看一区二区 | www..com毛片 | 丁香一区二区 | 中文字幕av影院 | 在线蜜桃视频 | 国产精品美女久久久久久免费 | 国产综合激情 | 日韩精品视频免费专区在线播放 | 日本久久成人中文字幕电影 | 91亚洲精品国偷拍 | 亚洲综合色激情五月 | 麻豆视频免费播放 | 欧美a级片免费看 | 婷婷久久综合九色综合 | 亚洲精品黄网站 | 四虎永久国产精品 | 日本aa在线 | 综合在线亚洲 | 激情图片区 | 亚洲成人影音 | 色偷偷88888欧美精品久久久 | 成人av网址大全 | 欧美在线资源 | 亚色视频在线观看 | 久久手机精品视频 | 亚洲综合视频在线 | 国产精品久久久久999 | 欧美一二三区播放 | 日韩免费一级a毛片在线播放一级 | 天天干天天射天天爽 | 中文字幕日韩无 | 精品综合久久 | 九九九热精品免费视频观看 | 在线国产视频 | 国产精品久久久久久久久久新婚 | 99久久久国产精品免费99 | 狠狠操91| 日韩欧美xxx | 亚洲激情校园春色 | 狠狠躁天天躁 | 成人精品一区二区三区中文字幕 | 日韩欧美有码在线 | 99精品电影| 国产午夜影院 | 中文字幕在线播放日韩 | 91视频在线免费下载 | 亚洲精品国产免费 | 美女视频黄免费网站 | 欧美激情精品久久久久久免费印度 | 久久精品影视 | 亚洲一区二区天堂 | 色婷婷av一区二 | 久草网在线视频 | 夜夜躁日日躁狠狠久久88av | 国产日韩欧美在线观看视频 | 成人免费网站在线观看 | 国产精品久久久久久久久久99 | 久久综合色综合88 | 日韩美在线观看 | 久久无码精品一区二区三区 | 精品久久久久久久 | 久久精品99精品国产香蕉 | 91视频xxxx| 亚洲免费小视频 | 久久久精品在线观看 | 狂野欧美激情性xxxx | 亚洲a在线观看 | 国产成人三级一区二区在线观看一 | 国产一级不卡视频 | 丁香花中文在线免费观看 | 成人av片免费看 | 激情欧美丁香 | 欧女人精69xxxxxx| 五月婷婷久 | 日韩欧美一区二区在线观看 | 国产精品原创在线 | 久草香蕉在线视频 | 91av网址| 婷婷干五月| 国产成人三级 | 精品国产亚洲一区二区麻豆 | 午夜a区 | 91亚洲精品视频 | 99久久久久久久久久 | 国产亚洲精品久久久久久电影 | 在线观看国产高清视频 | 91在线视频免费 | 最近日本字幕mv免费观看在线 | 日本护士三级少妇三级999 | 久久网页 | 日韩精品一区二区三区不卡 | 五月天,com| 69国产盗摄一区二区三区五区 | 韩国精品视频在线观看 | 中文字幕国产精品一区二区 | 在线播放国产一区二区三区 | 中文字幕免费在线看 | 狠狠狠狠狠狠狠 | 日本mv大片欧洲mv大片 | 亚洲精品久久久久久久不卡四虎 | 天天操天天摸天天爽 | 日韩一区正在播放 | 国产一区二三区好的 | 免费视频区 | 中文字幕在线国产 | 亚洲精品在线观看不卡 | 久久综合久久久久88 | 国产色网站 | 亚洲aⅴ免费在线观看 | 在线播放国产一区二区三区 | 久久久精品国产免费观看一区二区 | 久久综合久久综合久久综合 | 狠狠色噜噜狠狠狠狠2022 | 国产91综合一区在线观看 | av免费在线播放 | 超碰人人乐 | 日韩欧美精品在线视频 | 国产精品高清在线观看 | 天天操天天怕 | 亚洲成人欧美 | 亚洲四虎在线 | 成人a视频| 日韩av进入| 国产亚洲欧美一区 | a色视频 | 国产精品成人久久久久 | 国产一区视频在线播放 | 在线播放第一页 | 激情综合电影网 | 亚洲精品在线播放视频 | 国产黄免费看 | 黄色福利视频网站 | 又黄又爽又刺激 | 东方av免费在线观看 | 97人人模人人爽人人喊中文字 | 成人资源在线播放 | 操操操综合 | 亚洲精品福利在线 | 欧美日韩一区二区三区免费视频 | 久久在线精品视频 | 91免费观看国产 | 欧美精品中文在线免费观看 | 婷婷丁香色综合狠狠色 | 免费看片日韩 | 日韩 国产 | 国产美女视频黄a视频免费 久久综合九色欧美综合狠狠 | 国产专区在线看 | 亚洲精品视频免费在线 | 蜜臀av免费一区二区三区 | 9在线观看免费高清完整 | 亚洲黄色免费在线 | 久久久这里有精品 | 久久久毛片| 日韩视频1区 | 97成人在线免费视频 | 国产精品久久在线 | 天天狠狠| 亚洲不卡av一区二区三区 | 精品一区91 | 伊人日日干 | 欧美日韩高清在线一区 | 在线播放国产精品 | 欧女人精69xxxxxx | 精品欧美在线视频 | 国产精品第72页 | 久99精品| 日韩av在线看 | 日本一区二区不卡高清 | 我爱av激情网 | 五月天中文字幕mv在线 | 精品国产精品国产偷麻豆 | 狠狠色综合网站久久久久久久 | 色.www | 免费观看一级一片 | 婷婷视频导航 | 特级大胆西西4444www | 丁香婷婷激情国产高清秒播 | 久久久久国产精品免费免费搜索 | 日韩综合色 | 色综合久久88色综合天天免费 | 在线之家免费在线观看电影 | 欧美一级片在线 | 国产成人精品电影久久久 | 亚洲欧美成aⅴ人在线观看 四虎在线观看 | av在线色| 97精品在线视频 | www.夜夜草| 久草在线在线视频 | 69精品视频在线观看 | 欧美精品乱码久久久久久按摩 | 久久五月婷婷丁香 | 国产精品激情偷乱一区二区∴ | 亚洲视频中文 | 久久精品一区二区三区视频 | 夜夜爽天天爽 | 久久天天操 | 九九热免费在线观看 | 国产一区二区在线播放 | 久久精品国亚洲 | 久草新在线| 成人免费观看完整版电影 | 成人福利av | 9999免费视频 | 国产精品久久久久久久久久新婚 | 天天干天天干天天干天天干天天干天天干 | 天天爽综合网 | 97色国产| 国产精品18久久久久久久久久久久 | 中文字幕一区二区三区四区视频 | 人人爽人人爽人人爽人人爽 | 91亚洲精品久久久中文字幕 | 免费www视频 | 免费的黄色av | 日韩欧美一区二区三区视频 | 少妇做爰k8经典 | 国产精品成人久久久久 | 狠狠综合久久 | 日韩久久一区二区 | 日韩在线视频网 | 日韩在线激情 | 免费av一级电影 | 国产黄 | 国产精品99久久免费黑人 | 国产精品区一区 | 免费黄色小网站 | 欧美网址在线观看 | 久久精品视频网 | 亚洲精品国产综合久久 | 91电影福利 | 91av视频 | 国产成人一区二区在线观看 | 天天射天天爱天天干 | 久久综合福利 | 免费在线观看国产黄 | 久久激情五月丁香伊人 | 又黄又刺激视频 | 在线观看亚洲专区 | 久久精品精品 | 狠狠操狠狠干2017 | 亚洲黄色网络 | 97超碰资源总站 | 日本三级久久 | 国产精品一区二区三区在线播放 | 在线看成人av | 国产精品精品视频 | 97视频免费在线观看 | 日韩国产在线观看 | 成人黄色大片网站 | 亚洲欧美视频网站 | 久久久久二区 | 亚洲精品国精品久久99热 | 综合久久网 | 国产精品麻豆欧美日韩ww | 高清av免费观看 | x99av成人免费 | 一二区精品 | 免费欧美精品 | 欧美日韩精品久久久 | www五月婷婷 | a√天堂中文在线 | 丁香五香天综合情 | 欧美另类重口 | 免费黄色看片 | 中文在线字幕免费观 | 亚洲欧洲日韩在线观看 | 国产精品一区二区av麻豆 | 日韩色在线观看 | 91视频亚洲 | 国产一区二区视频在线播放 | 欧美va天堂在线电影 | 最近免费在线观看 | 亚洲成人一二三 | 一级黄色片在线 | 在线免费亚洲 | 中日韩男男gay无套 日韩精品一区二区三区高清免费 | 亚洲人成影院在线 | 欧美在线观看视频 | 91网免费观看 | 五月天天av| 久草香蕉在线视频 | 色视频在线免费观看 | 国产午夜精品一区 | 日韩av免费大片 | 在线观看片 | 久久综合狠狠综合 | 奇米影视777影音先锋 | 日韩三级一区 | 亚洲成人免费在线观看 | 欧美九九九 | 又黄又爽又刺激的视频 | 欧美精品日韩 | 免费日韩一区 | 激情五月婷婷综合 | 日韩精品一区二区三区视频播放 | 成人超碰在线 | 日本丰满少妇免费一区 | 日韩av电影中文字幕在线观看 | 99爱在线观看 | 午夜av在线 | 少妇视频在线播放 | 欧美性色黄大片在线观看 | 日韩久久一区二区 | 色天天综合网 | 久久久久激情视频 | 亚洲精品视频大全 | 久草在线网址 | 国产精品久久久久久久久久久久午夜片 | 亚洲成人黄色在线观看 | 在线看成人av | 天天干天天射天天插 | 又黄又爽又刺激的视频 | 亚洲欧美日韩国产精品一区午夜 | 精品成人国产 | 爱爱av网 | 国产一级做a爱片久久毛片a | 亚洲精品视频在线免费 | 午夜精品视频一区 | 天堂av高清 | 欧美aaaxxxx做受视频 | 中文字幕色婷婷在线视频 | 4438全国亚洲精品观看视频 | 91在线视频在线 | 国内精品久久久久影院男同志 | 天天综合区 | 999久久国产 | 久久精品精品电影网 | 久草视频免费观 | 欧美日韩一级久久久久久免费看 | 国产 日韩 欧美 中文 在线播放 | 青青久草在线视频 | 国产精品手机在线观看 | 24小时日本在线www免费的 | 国产久视频 | 亚洲亚洲精品在线观看 | 亚洲午夜久久久久久久久久久 | 国产精品一区二区精品视频免费看 | 久99久在线视频 | 99中文字幕视频 | av最新资源 | 很污的网站 | 亚洲高清精品在线 | 九色91福利| www.久久色.com | 亚洲电影成人 | 六月丁香激情综合色啪小说 | 国产精品精品国产色婷婷 | 99久久精品午夜一区二区小说 | 久久久久99精品国产片 | 日日碰狠狠躁久久躁综合网 | 中文字幕有码在线观看 | 精品亚洲成人 | 六月丁香激情网 | 91香蕉视频 mp4| 欧美少妇18p| 亚洲免费精品视频 |