


Beberapa fungsi yang berkaitan dengan pengurusan memori linux
Fungsi berkaitan pengurusan memori Linux: 1. kmalloc(), digunakan untuk peruntukan memori dalam mod kernel 2. vmalloc(), biasanya digunakan untuk fungsi yang hanya wujud dalam perisian (tanpa makna perkakasan yang sepadan) Urutan yang lebih besar; penampan memperuntukkan memori;
Persekitaran pengendalian tutorial ini: sistem linux7.3, komputer Dell G3.
Artikel ini menerangkan beberapa fungsi peruntukan memori biasa dalam kernel Linux dan persamaan dan perbezaannya, supaya mempunyai pemahaman yang lebih baik tentang mekanisme peruntukan memori asas Linux.
1. peruntukan memori, yang terakhir digunakan dalam mod pengguna. Fungsi kmalloc() memperuntukkan ruang storan
berterusan dalam memori fizikal Seperti fungsi malloc(), ia tidak akan mengosongkan data asal di dalamnya Jika memori mencukupi, kelajuan peruntukannya adalah sangat cepat. Prototaipnya adalah seperti berikut:static inline void *kmalloc(size_t size, gfp_t flags); /*返回的是虚拟地址*/
- bendera: Parameter ini digunakan untuk mengawal kelakuan fungsi yang paling biasa digunakan ialah GFP_KERNEL, yang bermaksud apabila tiada memori yang diperuntukkan pada masa ini, proses pergi ke. tidur dan tunggu sistem menyimpan kandungan penimbal Selepas SWAP ke cakera keras, dapatkan memori yang mencukupi dan kemudian bangunkan proses untuk memperuntukkannya. Lihat gambar di bawah untuk lebih banyak bendera:
-
Apabila menggunakan bendera GFP_ KERNEL untuk memohon ingatan, jika tidak dapat dipuaskan buat sementara waktu, proses akan tidur dan menunggu halaman, yang akan menyebabkan sekatan, jadi ia tidak boleh digunakan dalam konteks gangguan atau Gunakan GFP_KERNE untuk memohon memori apabila memegang kunci putaran. Oleh itu, ia tidak boleh menyekat dalam konteks bukan proses seperti pengendali gangguan, tugasan dan pemasa kernel Dalam kes ini, pemacu harus menggunakan bendera GFP_ATOMIC untuk memohon memori. Apabila menggunakan bendera GFP_ATOMIC untuk memohon ingatan, jika tiada halaman percuma, ia akan kembali terus tanpa menunggu.
- :
- _ _GFP_DMA (diperlukan untuk diperuntukkan dalam kawasan memori berkeupayaan DMA) _ _GFP_HIGHMEM (Menunjukkan bahawa memori yang diperuntukkan boleh ditempatkan dalam memori tinggi)
- _ _GFP_COLD (Meminta halaman yang sudah lama tidak diakses)
- _ _GFP_NOWARN (Menghalang kernel daripada mengeluarkan amaran apabila peruntukan tidak dapat dipenuhi)
- _ _GFP_HIGH (permintaan keutamaan tinggi, membolehkan untuk mendapatkan halaman memori terakhir yang dikhaskan oleh kernel untuk kegunaan kecemasan)
- _ _GFP_REPEAT (cuba semula usaha terbaik jika peruntukan gagal)
- _ _GFP_NOFAIL (tanda hanya membenarkan permohonan yang berjaya, tidak digalakkan)
- _ _GFP_NORETRY (jika permohonan tidak boleh dipohon, menyerah segera)
- Memori yang digunakan untuk menggunakan kmalloc() hendaklah dikeluarkan menggunakan , penggunaan fungsi ini adalah serupa dengan free() dalam ruang pengguna.
kfree()
biasanya digunakan kerana ia hanya wujud dalam perisian (tidak mempunyai makna perkakasan yang sepadan ) untuk memperuntukkan memori daripada penimbal jujukan yang lebih besar Apabila memori tidak mempunyai ruang fizikal berterusan yang cukup besar untuk diperuntukkan, fungsi ini boleh digunakan untuk memperuntukkan memori dengan alamat maya berterusan tetapi alamat fizikal bukan berterusan . Memandangkan jadual halaman baharu perlu dibuat, overhednya jauh lebih besar daripada kmalloc dan fungsi
yang akan dibincangkan kemudian. Dan tidak boleh digunakan dalam konteks atom kerana pelaksanaan dalamannya menggunakan 在linux中,内存分配是以页为单位的,32位机中一页为4KB,64位机中,一页为8KB,但具体还有根据平台而定。 根据返回页面数目分类,分为仅返回单页面的函数和返回多页面的函数。 该函数定义在头文件 它是kmalloc函数实现的基础,返回一个或多个页面的虚拟地址。该系列函数/宏包括 free_pages()函数是调用__free_pages()函数完成内存释放的。 该函数创建一个slab缓存(后备高速缓冲区),它是一个可以驻留任意数目全部同样大小的后备缓存。其原型如下: 其中: 一旦创建完后备高速缓冲区后,就可以调用 cachep指向开始分配的后备高速缓存,flags与传给kmalloc函数的参数相同,一般为GFP_KERNEL。 该函数释放一个内存块对象: 与 它必须等待所有已经分配的内存块对象被释放后才能释放后备高速缓存区。 创建一个存放线程结构体(struct thread_info)的后备高速缓存,因为在linux中涉及频繁的线程创建与释放,如果使用__get_free_page()函数会造成内存的大量浪费,效率也不高。所以在linux内核的初始化阶段就创建了一个名为thread_info的后备高速缓存,代码如下: 在 Linux 内核中还包含对内存池的支持,内存池技术也是一种非常经典的用于分配大量小对象的后备缓存技术。 mempool_create()函数用于创建一个内存池,min_nr 参数是需要预分配对象的数目,alloc_fn 和 free_fn 是指向内存池机制提供的标准对象分配和回收函数的指针,其原型分别为: pool_data 是分配和回收函数用到的指针,gfp_mask 是分配标记。只有当 在内存池中分配和回收对象需由以下函数来完成: mempool_alloc()用来分配对象,如果内存池分配器无法提供内存,那么就可以用预分配的池。 mempool_create()函数创建的内存池需由 mempool_destroy()来回收。 相关推荐:《Linux视频教程》vmalloc()
dengan bendera GFP_KERNEL. Prototaip fungsinya adalah seperti berikut: void *vmalloc(unsigned long size);
void vfree(const void *addr);
create_module()
系统调用,它利用 vmalloc()函数来获取被创建模块需要的内存空间。copy_from_user()
对内存进行使用。下面举一个使用vmalloc函数的示例:static int xxx(...)
{
...
cpuid_entries = vmalloc(sizeof(struct kvm_cpuid_entry) * cpuid->nent);
if(!cpuid_entries)
goto out;
if(copy_from_user(cpuid_entries, entries, cpuid->nent * sizeof(struct kvm_cpuid_entry)))
goto out_free;
for(i=0; i<cpuid->nent; i++){
vcpuid->arch.cpuid_entries[i].eax = cpuid_entries[i].eax;
...
vcpuid->arch.cpuid_entries[i].index = 0;
}
...
out_free:
vfree(cpuid_entries);
out:
return r;
}
3、页分配函数
根据返回值类型的不同,页分配函数分为两类,一是返回物理页地址,二是返回虚拟地址。虚拟地址和物理地址起始相差一个固定的偏移量。#define __pa(x) ((x) - PAGE_OFFSET)
static inline unsigned long virt_to_phys(volatile void *address)
{
return __pa((void *)address);
}
#define __va(x) ((x) + PAGE_OFFSET)
static inline void* phys_to_virt(unsigned long address)
{
return __va(address);
}
3.1 alloc_page()和alloc_pages()函数
/include/linux/gfp.h
中,它既可以在内核空间分配,也可以在用户空间分配,它返回分配的第一个页的描述符而非首地址,其原型为:#define alloc_page(gfp_mask) alloc_pages(gfp_mask, 0)
#define alloc_pages(gfp_mask, order) alloc_pages_node(numa_node_id(), gfp_mask, order) //分配连续2^order个页面
static inline struct page *alloc_pages_node(int nid, gfp_t gfp_mask, unsigned int order)
{
if(unlikely(order >= MAX_ORDER))
return NULL;
if(nid < 0)
nid = numa_node_id();
return __alloc_pages(gfp_mask, order, noed_zonelist(nid, gfp_mask));
}
3.2 __get_free_pages()系列函数
get_zeroed_page()
、_ _get_free_page()
和_ _get_free_pages()
。在使用时,其申请标志的值及含义与 kmalloc()
完全一样,最常用的是 GFP_KERNEL 和 GFP_ATOMIC。/*分配多个页并返回分配内存的首地址,分配的页数为2^order,分配的页不清零。
order 允许的最大值是 10(即 1024 页)或者 11(即 2048 页),依赖于具体
的硬件平台。*/
unsigned long __get_free_pages(gfp_t gfp_mask, unsigned int order)
{
struct page *page;
page = alloc_pages(gfp_mask, order);
if(!page)
return 0;
return (unsigned long)page_address(page);
}
#define __get_free_page(gfp_mask) __get_free_pages(gfp_mask, 0)
/*该函数返回一个指向新页的指针并且将该页清零*/
unsigned long get_zeroed_page(unsigned int flags);
_ _get_free_pages()
系列函数/宏申请的内存应使用free_page(addr)
或free_pages(addr, order)
函数释放:#define __free_page(page) __free_pages((page), 0)
#define free_page(addr) free_pages((addr), 0)
void free_pages(unsigned long addr, unsigned int order)
{
if(addr != 0){
VM_BUG_ON(!virt_addr_valid((void*)addr));
__free_pages(virt_to_page((void *)addr), order);
}
}
void __free_pages(struct page *page, unsigned int order)
{
if(put_page_testzero(page)){
if(order == 0)
free_hot_page(page);
else
__free_pages_ok(page, order);
}
}
4、slab缓存
kmem_cache
的结构体。4.1 创建slab缓存区
struct kmem_cache *kmem_cache_create(const char *name, size_t size, \
size_t align, unsigned long flags,\
void (*ctor)(void *, struct kmem_cache *, unsigned long),\
void (*dtor)(void *, struct kmem_cache *, unsigned ong)));
name:创建的缓存名;
size:可容纳的缓存块个数;
align:后备高速缓冲区中第一个内存块的偏移量(一般置为0);
flags:控制如何进行分配的位掩码,包括 SLAB_NO_REAP
(即使内存紧缺也不自动收缩这块缓存)、SLAB_HWCACHE_ALIGN
( 每 个 数 据 对 象 被 对 齐 到 一 个 缓 存 行 )、SLAB_CACHE_DMA
(要求数据对象在 DMA 内存区分配)等);
ctor:是可选的内存块对象构造函数(初始化函数);
dtor:是可选的内存对象块析构函数(释放函数)。4.2 分配slab缓存函数
kmem_cache_alloc()
在缓存区分配一个内存块对象了,其原型如下:void *kmem_cache_alloc(struct kmem_cache *cachep, gfp_t flags);
4.3 释放slab缓存
void *kmem_cache_free(struct kmem_cache *cachep, void *objp);
4.4 销毁slab缓存
kmem_cache_create
对应的是销毁函数,释放一个后备高速缓存:int kmem_cache_destroy(struct kmem_cache *cachep);
4.5 slab缓存使用举例
/* 创建slab缓存 */
static struct kmem_cache *thread_info_cache;
thread_info_cache = kmem_cache_create("thread_info", sizeof(struct thread_info), \
SLAB_HWCACHE_ALIGN|SLAB_PANIC, NULL, NULL);
/* 分配slab缓存 */
struct thread_info *ti;
ti = kmem_cache_alloc(thread_info_cache, GFP_KERNEL);
/* 使用slab缓存 */
...
/* 释放slab缓存 */
kmem_cache_free(thread_info_cache, ti);
kmem_cache_destroy(thread_info_cache);
5、内存池
5.1 创建内存池
mempool_t *mempool_create(int min_nr, mempool_alloc_t *alloc_fn, \
mempool_free_t *free_fn, void *pool_data);
typedef void *(mempool_alloc_t)(int gfp_mask, void *pool_data);
typedef void (mempool_free_t)(void *element, void *pool_data);
_ _GFP_WAIT
标记被指定时,分配函数才会休眠。5.2 分配和回收对象
void *mempool_alloc(mempool_t *pool, int gfp_mask);
void mempool_free(void *element, mempool_t *pool);
5.3 销毁内存池
void mempool_destroy(mempool_t *pool);
Atas ialah kandungan terperinci Beberapa fungsi yang berkaitan dengan pengurusan memori linux. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Alat AI Hot

Undresser.AI Undress
Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover
Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Undress AI Tool
Gambar buka pakaian secara percuma

Clothoff.io
Penyingkiran pakaian AI

AI Hentai Generator
Menjana ai hentai secara percuma.

Artikel Panas

Alat panas

Notepad++7.3.1
Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina
Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1
Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6
Alat pembangunan web visual

SublimeText3 versi Mac
Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Topik panas



Perbezaan utama antara CentOS dan Ubuntu adalah: asal (CentOS berasal dari Red Hat, untuk perusahaan; Ubuntu berasal dari Debian, untuk individu), pengurusan pakej (CentOS menggunakan yum, yang memberi tumpuan kepada kestabilan; Ubuntu menggunakan APT, untuk kekerapan yang tinggi) Pelbagai tutorial dan dokumen), kegunaan (CentOS berat sebelah ke arah pelayan, Ubuntu sesuai untuk pelayan dan desktop), perbezaan lain termasuk kesederhanaan pemasangan (CentOS adalah nipis)

Langkah Pemasangan CentOS: Muat turun Imej ISO dan Burn Bootable Media; boot dan pilih sumber pemasangan; Pilih susun atur bahasa dan papan kekunci; Konfigurasikan rangkaian; memisahkan cakera keras; Tetapkan jam sistem; Buat pengguna root; pilih pakej perisian; Mulakan pemasangan; Mulakan semula dan boot dari cakera keras selepas pemasangan selesai.

CentOS akan ditutup pada tahun 2024 kerana pengedaran hulu, RHEL 8, telah ditutup. Penutupan ini akan menjejaskan sistem CentOS 8, menghalangnya daripada terus menerima kemas kini. Pengguna harus merancang untuk penghijrahan, dan pilihan yang disyorkan termasuk CentOS Stream, Almalinux, dan Rocky Linux untuk memastikan sistem selamat dan stabil.

Dasar sandaran dan pemulihan Gitlab di bawah sistem CentOS untuk memastikan keselamatan data dan pemulihan, Gitlab pada CentOS menyediakan pelbagai kaedah sandaran. Artikel ini akan memperkenalkan beberapa kaedah sandaran biasa, parameter konfigurasi dan proses pemulihan secara terperinci untuk membantu anda menubuhkan strategi sandaran dan pemulihan GitLab lengkap. 1. Backup Manual Gunakan Gitlab-Rakegitlab: Backup: Buat Perintah untuk Melaksanakan Backup Manual. Perintah ini menyokong maklumat utama seperti repositori Gitlab, pangkalan data, pengguna, kumpulan pengguna, kunci, dan kebenaran. Fail sandaran lalai disimpan dalam direktori/var/opt/gitlab/sandaran. Anda boleh mengubah suai /etc /gitlab

Docker menggunakan ciri -ciri kernel Linux untuk menyediakan persekitaran berjalan yang cekap dan terpencil. Prinsip kerjanya adalah seperti berikut: 1. Cermin digunakan sebagai templat baca sahaja, yang mengandungi semua yang anda perlukan untuk menjalankan aplikasi; 2. Sistem Fail Kesatuan (Unionfs) menyusun pelbagai sistem fail, hanya menyimpan perbezaan, menjimatkan ruang dan mempercepatkan; 3. Daemon menguruskan cermin dan bekas, dan pelanggan menggunakannya untuk interaksi; 4. Ruang nama dan cgroups melaksanakan pengasingan kontena dan batasan sumber; 5. Pelbagai mod rangkaian menyokong interkoneksi kontena. Hanya dengan memahami konsep -konsep teras ini, anda boleh menggunakan Docker dengan lebih baik.

Centos Hard Disk Mount dibahagikan kepada langkah -langkah berikut: Tentukan nama peranti cakera keras (/dev/sdx); Buat titik gunung (disyorkan untuk menggunakan /mnt /newdisk); laksanakan perintah gunung (mount /dev /sdx1 /mnt /newdisk); Edit fail /etc /fstab untuk menambah konfigurasi gunung tetap; Gunakan perintah UMount untuk menyahpasang peranti untuk memastikan bahawa tiada proses menggunakan peranti.

Bagaimana cara menggunakan desktop Docker? Docktop Docktop adalah alat untuk menjalankan bekas Docker pada mesin tempatan. Langkah -langkah untuk digunakan termasuk: 1. Pasang desktop Docker; 2. Mulakan desktop Docker; 3. Buat imej Docker (menggunakan Dockerfile); 4. Membina imej Docker (menggunakan Docker Build); 5. Jalankan bekas Docker (menggunakan Docker Run).

Selepas CentOS dihentikan, pengguna boleh mengambil langkah -langkah berikut untuk menanganinya: Pilih pengedaran yang serasi: seperti Almalinux, Rocky Linux, dan CentOS Stream. Berhijrah ke pengagihan komersial: seperti Red Hat Enterprise Linux, Oracle Linux. Menaik taraf ke CentOS 9 Stream: Pengagihan Rolling, menyediakan teknologi terkini. Pilih pengagihan Linux yang lain: seperti Ubuntu, Debian. Menilai pilihan lain seperti bekas, mesin maya, atau platform awan.
