当前位置: 首页 > news >正文

C语言预读取技术 __builtin_prefetch

__builtin_prefetch 是一个编译器内置函数,用于在编译时向编译器发出指令,要求在执行期间预取内存数据。它通常用于提高程序的性能,特别是对于那些需要频繁访问内存的情况。


__builtin_prefetch 函数的语法如下:c
__builtin_prefetch(const void *ptr, int rw, int locality);
参数说明:ptr:一个指向要预取内存数据的指针。
rw:一个表示访问类型的整数。0 表示只读访问,1 表示读写访问。
locality:一个表示数据局部性的整数。0 表示没有局部性,1 表示数据访问是顺序的,2 表示数据访问是随机和独立的。

__builtin_prefetch 函数告诉编译器在执行期间预取 ptr 指向的内存数据,以便在后续的内存访问中可以更快地完成。通过指定 rw 参数,可以告诉编译器预取的数据是只读的还是有写操作的。最后,通过指定 locality 参数,可以告诉编译器预取的数据的局部性,以便编译器做出更明智的预取决策。

需要注意的是,__builtin_prefetch 函数是一个编译器内置函数,不是标准C语言的一部分。因此,它的可用性和具体实现可能因编译器而异。在使用时,建议查阅所使用编译器的文档以了解更多细节和用法

预读操作之所以能够生效,主要是因为现代计算机系统中的内存访问模式和硬件优化。

首先,计算机系统通常采用一种称为“缓存”的机制来优化内存访问。缓存是计算机内存中的一小部分,可以快速访问数据。CPU可以直接与缓存交互,而不需要通过相对较慢的主内存。当程序需要访问的数据不在缓存中时,这些数据将被从主内存加载到缓存中,以供后续访问。

然而,加载数据到缓存中需要一定的时间。为了最大限度地减少CPU等待数据的时间,现代计算机系统采用了一种称为“预读”的技术。预读是一种预测程序将要访问的数据并将其提前加载到缓存中的技术。通过预读,计算机系统可以在程序实际需要访问数据之前将其加载到缓存中,从而减少了CPU等待数据的时间,提高了程序的性能。

总之,预读操作之所以能够生效,是因为计算机系统采用了缓存机制和预读技术来优化内存访问和提高程序性能。这些技术允许程序更快地访问数据,减少了CPU等待数据的时间,从而提高了程序的性能。

#include <stdio.h>
#include <stdlib.h>#define PAGE_SIZE 4096 // 页面大小为4KB// 定义一个结构体表示内存页
typedef struct {int data[PAGE_SIZE / sizeof(int)];
} Page;// 预读取函数,将下一页数据加载到缓存中
void prefetch(Page* ptr) {asm ("prefetcht0 %0\n" // 将数据预读到TLB中:: "r" (ptr));
}int main() {Page* ptr = (Page*)malloc(PAGE_SIZE); // 分配一页内存空间if (ptr == NULL) {printf("Failed to allocate memory.\n");return -1;}// 初始化数据for (int i = 0; i < PAGE_SIZE / sizeof(int); i++) {ptr->data[i] = i;}// 执行预读取操作,将下一页数据加载到缓存中prefetch(ptr + 1);// 访问预读取的数据,并进行一些操作for (int i = 0; i < PAGE_SIZE / sizeof(int); i++) {printf("%d ", ptr[i + 1].data); // 访问预读取的数据}printf("\n");free(ptr); // 释放内存空间return 0;
}

以下是一个更高级的C语言预读取代码示例,它使用了指针和结构体来实现预读取机制,并采用了多线程和循环优化:

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>#define PAGE_SIZE 4096 // 页面大小为4KB
#define THREAD_NUM 4   // 线程数为4
#define LOOP_NUM 100   // 循环次数为100// 定义一个结构体表示内存页
typedef struct {int data[PAGE_SIZE / sizeof(int)];
} Page;// 预读取函数,将下一页数据加载到缓存中
void prefetch(Page* ptr) {asm ("prefetcht0 %0\n" // 将数据预读到TLB中:: "r" (ptr));
}// 线程函数,执行预读取和数据访问操作
void* thread_func(void* arg) {Page* ptr = (Page*)arg;for (int i = 0; i < LOOP_NUM; i++) {// 执行预读取操作,将下一页数据加载到缓存中prefetch(ptr + 1);// 访问预读取的数据,并进行一些操作for (int j = 0; j < PAGE_SIZE / sizeof(int); j++) {printf("%d ", ptr[j + 1].data); // 访问预读取的数据}printf("\n");}return NULL;
}int main() {pthread_t threads[THREAD_NUM]; // 定义线程数组Page* ptrs[THREAD_NUM]; // 分配内存页数组for (int i = 0; i < THREAD_NUM; i++) {ptrs[i] = (Page*)malloc(PAGE_SIZE); // 分配一页内存空间if (ptrs[i] == NULL) {printf("Failed to allocate memory.\n");return -1;}}// 初始化数据for (int i = 0; i < THREAD_NUM; i++) {for (int j = 0; j < PAGE_SIZE / sizeof(int); j++) {ptrs[i]->data[j] = i + j;}}// 创建线程并执行预读取和数据访问操作for (int i = 0; i < THREAD_NUM; i++) {pthread_create(&threads[i], NULL, thread_func, (void*)ptrs[i]);}for (int i = 0; i < THREAD_NUM; i++) {pthread_join(threads[i], NULL); // 等待线程结束}// 释放内存空间for (int i = 0; i < THREAD_NUM; i++) {free(ptrs[i]);}return 0;
}
http://www.lryc.cn/news/257458.html

相关文章:

  • 自动驾驶学习笔记(十三)——感知基础
  • WLAN配置实验
  • java_web接收前端传的excel文件读取数据
  • 在Vue开发中v-if指令和v-show指令的使用介绍和区别及使用场景
  • Power Query是啥
  • 在k8s中部署nfs-client-provisioner
  • 23.12.10日总结
  • 持续集成交付CICD:通过API方式上传Nexus制品
  • Hadoop学习笔记(HDP)-Part.14 安装YARN+MR
  • reinforce 跑 CartPole-v1
  • 【VRTK】【VR开发】【Unity】13-攀爬
  • 华为OD机试真题-求幸存数之和-2023年OD统一考试(C卷)
  • python pyaudio实时读取音频数据并展示波形图
  • 【算法系列篇】递归、搜索和回溯(二)
  • Ubuntu下安装SDL
  • 创建vue项目:vue脚手架安装、vue-cli安装,vue ui界面创建vue工程(vue2/vue3),安装vue、搭建vue项目开发环境(保姆级教程二)
  • 【3】密评-物理和环境安全测评
  • 笨爸爸工房,我们在校园|“小鲁班”,铸未来
  • RPC 集群,gRPC 广播和组播
  • OpenSSL SSL_read: Connection was reset, errno 10054
  • 【springboot】整合redis和定制化
  • HarmonyOS鸿蒙操作系统架构开发
  • 共创共赢|美创科技获江苏移动2023DICT生态合作“产品共创奖”
  • 深度学习——第3章 Python程序设计语言(3.5 Python类和对象)
  • 【原创】【一类问题的通法】【真题+李6卷6+李4卷4(+李6卷5)分析】合同矩阵A B有PTAP=B,求可逆阵P的策略
  • 代码随想录算法训练营第六十天 | 84.柱状图中最大的矩形
  • C#结合JavaScript实现多文件上传
  • STM32——继电器
  • 性能监控体系:InfluxDB Grafana Prometheus
  • CS106L2023 and CS106B 环境配置(详细教程)