首页 > 编程语言 >NVIDIA-CUDA编程初探

NVIDIA-CUDA编程初探

时间:2022-12-05 12:07:32浏览次数:44  
标签:__ 0000596f -- NVIDIA helloworld int CUDA 初探 gpu


CUDA的全称是Compute Unified Device Architecture,是显卡厂商NVIDIA推出的运算平台,开发者可以使用C语言来编写CUDA代码,使用NVCC编译器可以在支持CUDA的GPU处理器上以高速运行。虽然AMD也做显卡,但是CUDA是老黄自家提出的标准,没带AMD一起玩儿,所以,提到基于CUDA的高性能计算,使用的都是Nvidia的显卡。


NVIDIA-CUDA编程初探_mpx

nvidia-smi命令枚举了系统中的所有显卡支持信息

nvcc工具是CUDA编译器,用nvcc -V 验证编译器是否可以工作:

NVIDIA-CUDA编程初探_CUDA_02

cuda编程

编辑helloworld.cu文件,编码内容:

#include <cuda_runtime.h>
#include <stdio.h>

int main(void)
{
printf("hellow world!\n");
return 0;
}

之后执行 nvcc helloworld.cu -o helloworld,并运行

NVIDIA-CUDA编程初探_#include_03

可以看到,运行程序后打印除了helloworld.

但是,这个程序用到显卡了吗?很遗憾,没有。如果非要用显卡做点什么的化,可以改成这个样子:

#include <cuda_runtime.h>
#include <stdio.h>

__global__ void kernel(void)
{

}

int main(void)
{
kernel<<<1,1>>>();

printf("hellow world!\n");
return 0;
}

我们定义了一个空函数送给GPU跑,函数是空函数,什么也没做,白嫖一下GPU就退出,编译并运行:

NVIDIA-CUDA编程初探_mpx_04

生成的helloworld文件是ELF格式的目标文件,与GCC产生的无异,可以通过objdump反编译一把:

NVIDIA-CUDA编程初探_NVCC_05

来看一下main函数的片段:

NVIDIA-CUDA编程初探_CUDA_06

粗略一看,首先给人的印象是NVCC不是一个人在战斗,毕竟我们的代码才短短几行,反编译后却有这么多条指令,而且貌似有些指令是没有出现在源码层面调用的。还能看出一点的就是源码是按照C++编译的,因为看到了明显的名字改编。

那就是编译器做的手脚咯,幸好我们有办法确认这一点,方式就是在nvcc编译的时候加上--verbose选项:

NVIDIA-CUDA编程初探_#include_07

#$ _NVVM_BRANCH_=nvvm
#$ _SPACE_=
#$ _CUDART_=cudart
#$ _HERE_=/usr/local/cuda-11.5/bin
#$ _THERE_=/usr/local/cuda-11.5/bin
#$ _TARGET_SIZE_=
#$ _TARGET_DIR_=
#$ _TARGET_DIR_=targets/x86_64-linux
#$ TOP=/usr/local/cuda-11.5/bin/..
#$ NVVMIR_LIBRARY_DIR=/usr/local/cuda-11.5/bin/../nvvm/libdevice
#$ LD_LIBRARY_PATH=/usr/local/cuda-11.5/bin/../lib::/usr/local/cuda-11.5/lib64
#$ PATH=/usr/local/cuda-11.5/bin/../nvvm/bin:/usr/local/cuda-11.5/bin:/home/caozilong/anaconda3/bin:/home/caozilong/anaconda3/condabin:/home/caozilong/.local/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/snap/bin:/usr/local/cuda-11.5/bin
#$ INCLUDES="-I/usr/local/cuda-11.5/bin/../targets/x86_64-linux/include"
#$ LIBRARIES= "-L/usr/local/cuda-11.5/bin/../targets/x86_64-linux/lib/stubs" "-L/usr/local/cuda-11.5/bin/../targets/x86_64-linux/lib"
#$ CUDAFE_FLAGS=
#$ PTXAS_FLAGS=
#$ gcc -D__CUDA_ARCH__=520 -D__CUDA_ARCH_LIST__=520 -E -x c++ -DCUDA_DOUBLE_MATH_FUNCTIONS -D__CUDACC__ -D__NVCC__ "-I/usr/local/cuda-11.5/bin/../targets/x86_64-linux/include" -D__CUDACC_VER_MAJOR__=11 -D__CUDACC_VER_MINOR__=5 -D__CUDACC_VER_BUILD__=50 -D__CUDA_API_VER_MAJOR__=11 -D__CUDA_API_VER_MINOR__=5 -D__NVCC_DIAG_PRAGMA_SUPPORT__=1 -include "cuda_runtime.h" -m64 "helloworld.cu" -o "/tmp/tmpxft_0000596f_00000000-9_helloworld.cpp1.ii"
#$ cicc --c++14 --gnu_version=70500 --display_error_number --orig_src_file_name "helloworld.cu" --orig_src_path_name "/home/caozilong/cuda/helloworld.cu" --allow_managed -arch compute_52 -m64 --no-version-ident -ftz=0 -prec_div=1 -prec_sqrt=1 -fmad=1 --include_file_name "tmpxft_0000596f_00000000-3_helloworld.fatbin.c" -tused --gen_module_id_file --module_id_file_name "/tmp/tmpxft_0000596f_00000000-4_helloworld.module_id" --gen_c_file_name "/tmp/tmpxft_0000596f_00000000-6_helloworld.cudafe1.c" --stub_file_name "/tmp/tmpxft_0000596f_00000000-6_helloworld.cudafe1.stub.c" --gen_device_file_name "/tmp/tmpxft_0000596f_00000000-6_helloworld.cudafe1.gpu" "/tmp/tmpxft_0000596f_00000000-9_helloworld.cpp1.ii" -o "/tmp/tmpxft_0000596f_00000000-6_helloworld.ptx"
#$ ptxas -arch=sm_52 -m64 "/tmp/tmpxft_0000596f_00000000-6_helloworld.ptx" -o "/tmp/tmpxft_0000596f_00000000-10_helloworld.sm_52.cubin"
#$ fatbinary -64 --cicc-cmdline="-ftz=0 -prec_div=1 -prec_sqrt=1 -fmad=1 " "--image3=kind=elf,sm=52,file=/tmp/tmpxft_0000596f_00000000-10_helloworld.sm_52.cubin" "--image3=kind=ptx,sm=52,file=/tmp/tmpxft_0000596f_00000000-6_helloworld.ptx" --embedded-fatbin="/tmp/tmpxft_0000596f_00000000-3_helloworld.fatbin.c"
#$ rm /tmp/tmpxft_0000596f_00000000-3_helloworld.fatbin
#$ gcc -D__CUDA_ARCH_LIST__=520 -E -x c++ -D__CUDACC__ -D__NVCC__ "-I/usr/local/cuda-11.5/bin/../targets/x86_64-linux/include" -D__CUDACC_VER_MAJOR__=11 -D__CUDACC_VER_MINOR__=5 -D__CUDACC_VER_BUILD__=50 -D__CUDA_API_VER_MAJOR__=11 -D__CUDA_API_VER_MINOR__=5 -D__NVCC_DIAG_PRAGMA_SUPPORT__=1 -include "cuda_runtime.h" -m64 "helloworld.cu" -o "/tmp/tmpxft_0000596f_00000000-5_helloworld.cpp4.ii"
#$ cudafe++ --c++14 --gnu_version=70500 --display_error_number --orig_src_file_name "helloworld.cu" --orig_src_path_name "/home/caozilong/cuda/helloworld.cu" --allow_managed --m64 --parse_templates --gen_c_file_name "/tmp/tmpxft_0000596f_00000000-6_helloworld.cudafe1.cpp" --stub_file_name "tmpxft_0000596f_00000000-6_helloworld.cudafe1.stub.c" --module_id_file_name "/tmp/tmpxft_0000596f_00000000-4_helloworld.module_id" "/tmp/tmpxft_0000596f_00000000-5_helloworld.cpp4.ii"
#$ gcc -D__CUDA_ARCH__=520 -D__CUDA_ARCH_LIST__=520 -c -x c++ -DCUDA_DOUBLE_MATH_FUNCTIONS "-I/usr/local/cuda-11.5/bin/../targets/x86_64-linux/include" -m64 "/tmp/tmpxft_0000596f_00000000-6_helloworld.cudafe1.cpp" -o "/tmp/tmpxft_0000596f_00000000-11_helloworld.o"
#$ nvlink -m64 --arch=sm_52 --register-link-binaries="/tmp/tmpxft_0000596f_00000000-7_helloworld_dlink.reg.c" "-L/usr/local/cuda-11.5/bin/../targets/x86_64-linux/lib/stubs" "-L/usr/local/cuda-11.5/bin/../targets/x86_64-linux/lib" -cpu-arch=X86_64 "/tmp/tmpxft_0000596f_00000000-11_helloworld.o" -lcudadevrt -o "/tmp/tmpxft_0000596f_00000000-12_helloworld_dlink.sm_52.cubin"
#$ fatbinary -64 --cicc-cmdline="-ftz=0 -prec_div=1 -prec_sqrt=1 -fmad=1 " -link "--image3=kind=elf,sm=52,file=/tmp/tmpxft_0000596f_00000000-12_helloworld_dlink.sm_52.cubin" --embedded-fatbin="/tmp/tmpxft_0000596f_00000000-8_helloworld_dlink.fatbin.c"
#$ rm /tmp/tmpxft_0000596f_00000000-8_helloworld_dlink.fatbin
#$ gcc -D__CUDA_ARCH_LIST__=520 -c -x c++ -DFATBINFILE="\"/tmp/tmpxft_0000596f_00000000-8_helloworld_dlink.fatbin.c\"" -DREGISTERLINKBINARYFILE="\"/tmp/tmpxft_0000596f_00000000-7_helloworld_dlink.reg.c\"" -I. -D__NV_EXTRA_INITIALIZATION= -D__NV_EXTRA_FINALIZATION= -D__CUDA_INCLUDE_COMPILER_INTERNAL_HEADERS__ "-I/usr/local/cuda-11.5/bin/../targets/x86_64-linux/include" -D__CUDACC_VER_MAJOR__=11 -D__CUDACC_VER_MINOR__=5 -D__CUDACC_VER_BUILD__=50 -D__CUDA_API_VER_MAJOR__=11 -D__CUDA_API_VER_MINOR__=5 -D__NVCC_DIAG_PRAGMA_SUPPORT__=1 -m64 "/usr/local/cuda-11.5/bin/crt/link.stub" -o "/tmp/tmpxft_0000596f_00000000-13_helloworld_dlink.o"
#$ g++ -D__CUDA_ARCH_LIST__=520 -m64 -Wl,--start-group "/tmp/tmpxft_0000596f_00000000-13_helloworld_dlink.o" "/tmp/tmpxft_0000596f_00000000-11_helloworld.o" "-L/usr/local/cuda-11.5/bin/../targets/x86_64-linux/lib/stubs" "-L/usr/local/cuda-11.5/bin/../targets/x86_64-linux/lib" -lcudadevrt -lcudart_static -lrt -lpthread -ldl -Wl,--end-group -o "helloworld"

现在总结一下CUDA编程的规则:

  • 核函数,在GPU上执行的函数通常成为核函数,如上面程序中的kernel函数。
  • 核函数一般通过标识符__global__修饰,通过<<<参数1,参数2>>>调用,用于说明内核函数中的线程数量,以及线程是如何组织的。
  • 以线程格(Grid)的形式组织,每个线程格有若干个线程块(block)组成,而每个线程块又由若干个线程(thread)组成。
  • 以Block为单位执行
  • 能在主机端代码中调用
  • 调用时必须声明内核函数的执行参数
  • 在编程时,必须先为kernel函数中用到的数组或者变量分配好足够的空间,再调用kernel函数,否则在GPU计算时会发生错误,例如越界或者报错,甚至导致蓝屏和死机。

CUDA的变成模型如下图所示:

NVIDIA-CUDA编程初探_CUDA_08

上面例子中,kernel函数恰好叫kernel是一种巧合,实际上你可以改成任何有意义的名字,只要按照CUDA要求的方式调用即可

#include <cuda_runtime.h>
#include <stdio.h>

__global__ void dummy(void)
{

}

int main(void)
{
dummy<<<1,1>>>();

printf("hellow world!\n");
return 0;
}

NVIDIA-CUDA编程初探_NVCC_09

对于上如上的例子,我们探究一下它的控制流是如何进行的,首先我们看到反编译文件中,首先main函数调用了_Z5dummyv

NVIDIA-CUDA编程初探_CUDA_10

 不难看出这个函数名是经过C++名字改编的,我们用c++filt工具将其还原:

NVIDIA-CUDA编程初探_CUDA_11

可以看到它就是dummy,我们继续追踪

NVIDIA-CUDA编程初探_#include_12

可以看到dummy调用了_Z23__device_stub__Z5dummyvv函数,继续追踪,发现执行流最终调用了_Z16cudaLaunchKernelIcE9cudaErrorPKT_4dim3S4_PPvmP11CUstream_st,而经过反命名后,发现它是名为cudaLaunchKernel的cuda函数,这个函数并非代码中显示调用的,而是NVCC工具链生成的,所以,顾名思义,很可能就是这句调用发起了对GPU控制流的交接。

NVIDIA-CUDA编程初探_#include_13

 __global__和__device__

__global__和__device__是函数修饰符,__global__表明被修饰的函数在设备上执行,但是在主机上调用,__device__,表示被修饰的函数在设备上执行,但是只能在其他__device__或者__global__函数中调用,说白了它只能在GPU中执行,并且被GPU中执行的函数调用。

新的例子

计算3+6等于几的例子

#include <cuda_runtime.h>
#include <stdio.h>

__global__ void add(int a, int b, int *c)
{
*c = a + b;
}

int main(void)
{
int c;
int *gpu_c;

cudaMalloc((void **)&gpu_c, sizeof(int));

add<<<1,1>>>(3,6,gpu_c);

cudaMemcpy(&c, gpu_c, sizeof(int), cudaMemcpyDeviceToHost);

cudaFree(gpu_c);

printf("3 + 9 eques %d.\n", c);

return 0;
}

NVIDIA-CUDA编程初探_mpx_14

我们得到了正确的计算结果。

我们稍微改一下程序,将计算过程改为循环计算,然后用nvidia-smi工具监视一下GPU的资源使用情况:

#include <cuda_runtime.h>
#include <stdio.h>

__global__ void add(int a, int b, int *c)
{
*c = a + b;
}

int main(void)
{
int c;
int *gpu_c;

while(1)
{
cudaMalloc((void **)&gpu_c, sizeof(int));

add<<<1,1>>>(3,6,gpu_c);

cudaMemcpy(&c, gpu_c, sizeof(int), cudaMemcpyDeviceToHost);

cudaFree(gpu_c);

printf("3 + 9 eques %d.\n", c);
}

return 0;
}

编译并运行:

NVIDIA-CUDA编程初探_mpx_15

运行过程中,使用watch -n 1 nvidia-smi命令监控GPU的资源变化情况,可以看到内存占用和GPU负载在不断的发生变化:

NVIDIA-CUDA编程初探_NVCC_16

复杂一些的例子:

下面的例子对两个列向量求算术平方和,循环进行M次,分别用CPU和GPU计算,最后对统计到的计算速度进行对比:

#include <cuda_runtime.h>
#include <device_launch_parameters.h>
#include <stdio.h>
#include <time.h>

#define N (1024 * 1024)
#define M (10000)
#define THREADS_PER_BLOCK (1024)

void cpu_vector_add(double *a, double *b, double *c, int n , int m)
{
int index, j;

for(index = 0; index < n; index ++)
{
for(j = 0; j < m; j ++)
{
c[index] = a[index] * a[index] + b[index] * b[index];
}
}

return;
}

__global__ void gpu_vector_add(double *a, double *b, double *c)
{
int j;
int index = blockIdx.x * blockDim.x + threadIdx.x;

for(j = 0; j < M; j ++)
{
c[index] = a[index] * a[index] + b[index] * b[index];
}
}

int main(void)
{
clock_t start, end;
double *a, *b, *c;

int size = N * sizeof(double);

a = (double *)malloc(size);
b = (double *)malloc(size);
c = (double *)malloc(size);

if(!a || !b || !c)
{
printf("%s line %d, fatal error,malloc buffer failure.\n", __func__, __LINE__);
return -1;
}

int j;
for(j = 0; j < N; j ++)
{
a[j] = b[j] = j;
c[j] = 0;
}

start = clock();
cpu_vector_add(a, b, c, N, M);

printf("[%d]=%f\n", 0, c[0]);
printf("[%d]=%f\n", N-1, c[N-1]);

end = clock();

float time_cpu_cost = ((float)(end-start))/CLOCKS_PER_SEC;
printf("CPU cost %f sectonds.\n", time_cpu_cost);

start = clock();

double *gpu_a, *gpu_b, *gpu_c;

cudaMalloc((void**)&gpu_a, size);
cudaMalloc((void**)&gpu_b, size);
cudaMalloc((void**)&gpu_c, size);

cudaMemcpy(gpu_a, a, size, cudaMemcpyHostToDevice);
cudaMemcpy(gpu_b, b, size, cudaMemcpyHostToDevice);

gpu_vector_add<<< (N + (THREADS_PER_BLOCK-1)) / THREADS_PER_BLOCK, THREADS_PER_BLOCK >>>(gpu_a, gpu_b, gpu_c);

cudaMemcpy(c, gpu_c, size, cudaMemcpyDeviceToHost);

printf("[%d]=%f\n", 0, c[0]);
printf("[%d]=%f\n", N-1, c[N-1]);

end = clock();

float time_gpu_cost = ((float)(end-start))/CLOCKS_PER_SEC;
printf("GPU cost %f sectonds.\n", time_gpu_cost);

float faster_than = time_cpu_cost/time_gpu_cost;
printf("GPU cost faster than CPU %f times.\n", faster_than);

//printf("a = %p, b = %p, c = %p, gpu_a = %p, gpu_b = %p, gpu_c = %p.\n", a, b, c, gpu_a, gpu_b, gpu_c);

free(a);
free(b);
free(c);
cudaFree(gpu_a);
cudaFree(gpu_b);
cudaFree(gpu_c);

return 0;
}

编译运行,查看计算结果:

nvcc helloworld.cu

NVIDIA-CUDA编程初探_mpx_17

经过实际测试,同样的计算量,我的红米本使用的MX250入门级显卡要比Intel(R) Core(TM) i7-8565U CPU @ 1.80GHz处理器快大约30倍,更别提那些用于数学计算的专业N卡了,这个数据让我对显卡的恐怖计算能力有了进一步的认识。

问题:

程序中存在两类指针,malloc分配的主存指针和cudaMalloc分配的显存指针,添加打印,打印出a,b,c,gpu_a,gpu_b,gpu_c的指针数值,根据打印来看,这些指针没有明显差别,难道cudaMalloc分配的并非是显存上的地址?或者显存和主存之间存在某种映射?不过可以确定的是,虽然指针的地址范围相似,但是不可以在主机代码中使用​​cudaMalloc()​​分配的指针进行主机内存读写操作(即不能进行解引用)。

NVIDIA-CUDA编程初探_#include_18

#include <cuda_runtime.h>
#include <device_launch_parameters.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>

#define N (1024 * 1024)
#define M (10000)
#define THREADS_PER_BLOCK (1024)

void cpu_vector_add(double *a, double *b, double *c, int n , int m)
{
int index, j;

for(index = 0; index < n; index ++)
{
for(j = 0; j < m; j ++)
{
c[index] = a[index] * a[index] + b[index] * b[index];
}
}

return;
}

__global__ void gpu_vector_add(double *a, double *b, double *c)
{
int j;
int index = blockIdx.x * blockDim.x + threadIdx.x;

for(j = 0; j < M; j ++)
{
c[index] = a[index] * a[index] + b[index] * b[index];
}
}

int main(void)
{
clock_t start, end;
double *a, *b, *c;

int size = N * sizeof(double);

a = (double *)malloc(size);
b = (double *)malloc(size);
c = (double *)malloc(size);

if(!a || !b || !c)
{
printf("%s line %d, fatal error,malloc buffer failure.\n", __func__, __LINE__);
return -1;
}

int j;
for(j = 0; j < N; j ++)
{
a[j] = b[j] = j;
c[j] = 0;
}

start = clock();
cpu_vector_add(a, b, c, N, M);

printf("[%d]=%f\n", 0, c[0]);
printf("[%d]=%f\n", N-1, c[N-1]);

end = clock();

float time_cpu_cost = ((float)(end-start))/CLOCKS_PER_SEC;
printf("CPU cost %f sectonds.\n", time_cpu_cost);

start = clock();

double *gpu_a, *gpu_b, *gpu_c;

cudaMalloc((void**)&gpu_a, size);
cudaMalloc((void**)&gpu_b, size);
cudaMalloc((void**)&gpu_c, size);

cudaMemcpy(gpu_a, a, size, cudaMemcpyHostToDevice);
cudaMemcpy(gpu_b, b, size, cudaMemcpyHostToDevice);

gpu_vector_add<<< (N + (THREADS_PER_BLOCK-1)) / THREADS_PER_BLOCK, THREADS_PER_BLOCK >>>(gpu_a, gpu_b, gpu_c);

cudaMemcpy(c, gpu_c, size, cudaMemcpyDeviceToHost);

printf("[%d]=%f\n", 0, c[0]);
printf("[%d]=%f\n", N-1, c[N-1]);

end = clock();

float time_gpu_cost = ((float)(end-start))/CLOCKS_PER_SEC;
printf("GPU cost %f sectonds.\n", time_gpu_cost);

float faster_than = time_cpu_cost/time_gpu_cost;
printf("GPU cost faster than CPU %f times.\n", faster_than);

printf("a = %p, b = %p, c = %p, gpu_a = %p, gpu_b = %p, gpu_c = %p.\n", a, b, c, gpu_a, gpu_b, gpu_c);

while(1)
{
printf("%s line %d, *a = %f.\n", __func__, __LINE__, *a);
printf("%s line %d, *gpu_c = %f.\n", __func__, __LINE__, *gpu_c);
}

free(a);
free(b);
free(c);
cudaFree(gpu_a);
cudaFree(gpu_b);
cudaFree(gpu_c);

return 0;
}

NVIDIA-CUDA编程初探_CUDA_19

和OPENCL的关系:

架构上,它们在同一层面,cuda和OpenCL都属于一种并行计算开发语言,这从CUDA使用的编译器和OpenCL虽然使用GCC编译HOST侧代码,但是端册代码却需编译一个文本CL程序文件,交给OpenCL API执行在线编译看出来,他们虽然都有吸取C的语法特点,但是异构加速核心这一块和CPU端的编译器是不共用的,关于OPENCL开发的例子,可以参考如下博客。

后来的测试:

半年后,重新安装系统,使用DARKNET环境安装CUDA和CUDANN之后:

 ./setup.sh -InstallCUDA

NVIDIA-CUDA编程初探_CUDA_20

再用老办法安装NVIDIA 驱动 

NVIDIA-CUDA编程初探_mpx_21

此时CUDA运行速率明显加快,原因未知。

总结

NVCC是NVIDIA提供的用于编译CUDA C程序的编译器,它会自动将.cu文件分为带有CUDA C语句的部分和不带CUDA C语句的部分,并将后者交给本地的C/C++编译器,当两部分文件都编译完毕,NVCC再将他们连接成可执行文件,默认的编译命令不带有任何参数,可以直接生成可执行文件。

KERNEL有三种编译方式,分别为:

1.静态编译,也叫离线编译,在执行前由C++、GCC或者VENDOR SPECIFIC的工具链进行编译。

2.动态编译,也叫运行时编译,在CPU运行过程中编译GPU KENREL代码。

3.运行时IR编译,执行前先i将KERNEL代码编译为IR,用户驱动集成COMPILER的BACKEND代码,在运行时再将IR编译为GPU上运行的指令,是1,2,两种情况的融合。

NVIDIA-CUDA编程初探_NVCC_22

NVIDIA-CUDA编程初探_NVCC_23

OPENCL可以看作是CUDA的一个保守发行版,对用户来说,如果要使用NVIDIA产品的最新特性,则需要使用CUDA而不是OPENCL。

NVIDIA-CUDA编程初探_mpx_24


结束! 

标签:__,0000596f,--,NVIDIA,helloworld,int,CUDA,初探,gpu
From: https://blog.51cto.com/u_15899439/5911823

相关文章

  • 亚马逊自研芯片-异构芯片-NVIDIA分析
    亚马逊自研芯片-异构芯片-NVIDIA分析参考文献链接https://mp.weixin.qq.com/s/frLCeSpu7v_BnlJ6BcCRMQhttps://mp.weixin.qq.com/s/duHY4NbXSBIk6nc572V8wwhttps://m.m......
  • python自动化办公初探之桌牌制作
    前言:开会用的桌牌,制作起来非常麻烦,要根据参会人员的不同,制作不同的桌牌。如果参会人员非常多,制作就变的更麻烦。通过python中的xlrd和docxtpl模块可以自动的快速生成桌牌,省......
  • deepin国产操作系统 nvidia-docker2 的安装
       ======================================  平时偶尔使用deepin系统,突然有个 nvidia-docker的程序需要运行,平时工作都是在用Ubuntu,所以对deepin安装docker......
  • Ubuntu22.04+Nvidia驱动+Cuda11.8+cudnn8.6
    Ubuntu22.04+Nvidia驱动+Cuda11.8一、准备环境ubuntu22.04nvidia显卡这里使用的是RTX3060已安装Python3.10二、安装pip3#安装sudoaptinstallpython3-pip#升级sudopi......
  • 【RSA加密】初探RSA并简单使用
    RSA简介,这里贴上一篇博客,讲的很详细,通俗易懂初步理解之后,下面是关于RSA的简单使用:这里贴上一篇优秀的前端加密,后端解密的博客,简单使用的话是可以了。看完上面两篇博客,就够用......
  • Ubuntu22.04安装CUDA深度学习环境&&cuda principle
    environment:neofetch&&uname-a|lolcatinstallnvidiaGPUdriver:sudoadd-apt-repositoryppa:graphics-drivers/ppa#加入官方ppa源sudoaptupdate#检查软件包......
  • 一个由tf1.6.0引发的故事|从CUDA到gcc配置,非root用户重装旧版本TF环境
    之前尝试复现学姐前几年的一个工作,但是因为框架有点古老而作罢。然鹅,自己的实验结果一直跑得十分奇怪,为了去学姐的代码中寻找参考,今天再次进行了尝试。我的需求是安装T......
  • 跨平台开发:PhoneGap移动开发框架初探
    原文发表在:http://publish.itpub.net/a2010/1008/1111/000001111212.shtml目前,随着Google的Android手机和苹果的iphone手机的逐渐普及,越来越多开发......
  • 从头开始进行CUDA编程:原子指令和互斥锁
    在前三部分中我们介绍了CUDA开发的大部分基础知识,例如启动内核来执行并行任务、利用共享内存来执行快速归并、将可重用逻辑封装为设备函数以及如何使用事件和流来组织和控......
  • Ubuntu下nvidia驱动卸载
    Ubuntu下nvidia驱动卸载的一种方法卸掉已经安装的驱动:sudoapt-getremove--purge'^nvidia-.*'sudoapt-getremove--purge'^libnvidia-.*'sudoapt-getremove--pu......