IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 系统运维 -> 虚拟地址向物理地址转换 笔记 -> 正文阅读

[系统运维]虚拟地址向物理地址转换 笔记

回顾相关概念

逻辑地址--->线性地址又称虚拟地址--->物理地址

代码段是只读的、数据段是可写的、堆栈段是可增长的

????????在进程中,我们不直接对物理地址进行操作。CPU在运行时,指定的地址要经过MMU转换后才能访问到真正的物理内存。地址转换的过程分为两部分,分段和分页。分段机制简单点来讲就是将进程的代码、数据、栈分在不同的虚拟地址段上。从而避免进程间的互相影响,分段之前的地址我们称之为逻辑地址。

????????逻辑地址由两部分组成,高位的段选择符和低位的段内偏移。在分段时先用段选择符在相应的段描述符表中找到段描述符,也就是某一个段的基地址,再加上段内偏移量就得到了相应的线性地址。在实际的应用中,Linux系统为了增加可移植性并没有完整的使用段机制,它让所有的段都指向相同的地址范围,由于段地址基址都为0,所以逻辑地址和线性地址数值上相同,这里主要分析分页也就是线性地址到物理地址的转换。

????????linux为了兼容32位和64位的CPU,它需要一个统一的页面地址模型,目前常用的是4级页表模型。4级页表分别包括:PGD页全局目录、PUD页上级目录、PMD页中间目录、PT页表。根据不同的需要,某些页表可能未被使用,线性地址也是每一部分的索引会根据具体的计算机体系结构来做出相应的改变,举例来讲,对于未启用物理地址的32位操作系统来说,两级页表就是足够的。Linux会首先在线性地址中将页上级目录和页中间目录索引这两位置为0,从根本上就取消了这两个字段,但是这两个页目录在指针序列中的位置仍然被保留了下来,也就是说寻址过程中不能跳过页上级目录和页中间目录直接由页全局目录直接到页表,内核会将这两个页目录的表项都置为1。

????????由于64系统硬件的限制,它的地址线只有48条,所以线性地址使用的只有48位,在64位Linux中使用了4级页表结构。首先页全局目录的索引、页上级目录的索引、页中间目录的索引、页表的索引分别是占了9位。

0-11??? 12位?? 分给页内偏移

11-20?? 9位??? 分给页表

20-29?? 9位??? 分给页中间目录

29-38?? 9位??? 分给页上级目录

38-47?? 9位??? 分给页全局目录

47-63?? 16位? 留作保留位以后使用

由于页内偏移是12位,则页面的大小是2^12=4k𝐵 (怎么进入和退出公式编辑模式,alt + =)

每一个页表项大小为8bit,整个页表可以映射的空间是256TB

每个页表项为什么是8bit?(未搞清楚,求大神指导)

2^48=256TB?

2的10次方 1KB

2的20次方 1MB

2的30次方 1GB

2的40次方 1TB

2的48次方 256TB

????????新的Intel芯片的MMU硬件规定可以进行5级的页表管理,所以在4.15的内核中,linux在页全局目录和页上级目录中间增加了一个新的页目录,叫p4d页目录。这个页目录同32位中的情况一样现在还未被使用,它的页目录项只有一个,线性地址中也没有它的索引位。

CR3寄存器是用来干什么的?

????????它是一系列CPU控制寄存器之一 ,这些控制寄存器主要是用来保存控制系统级别操作的数据和信息,其中这个CR3就是用来保存当前进程的页全局目录的地址的。寻页的开始就是从页全局目录开始。

页全局目录的地址又在哪里?怎么进行页表的切换?

????????内核在创建一个进程时就会为它分配页全局目录,在进程描述符task_struct结构中有一个指向mm_struct结构的指针mm,而mm_struct结构就是用来描述进程的虚拟地址空间的,在mm_struct中有一个字段PGD就是用来保存全局目录的(物理)地址的,所以在进程切换的时候,操作系统通过访问task_struct结构,在访问mm_struct结构,最终找到PGD字段,取得新进程的页全局目录的地址填充到CR3寄存器中完成页表的切换。

接下来进行实际操作,看看寻页过程是如何完成的。

????????可以看出PGD_SHIFT和P4D_SHIFT都是39,意味着在线性地址中,P4D这个字段是空的

在页目录项中,P4D的页目录项也是为1的,说明现在的Linux系统使用了五级的页表模型,

但实际上使用的页表也只有四个。

????????PAGE_MASK字段它是低12位都为0,其余位都为1的一个64位的数。

我们已经通过线性地址得到了物理地址,现在想通过直接访问物理地址看看里面存放的是什么??

????????需要dram.c,主要功能是用mmap将物理内存中的数据映射到设备文件中,通过对于这个设备文件的访问就可以达到访问物理内存的功能。

????????另一个工具是fileview.cpp,可以让我们按照想要的格式来阅读二进制文件。

????????现在使用这两个小工具对物理内存进行访问。

遇到错误 参考链接:(27条消息) make menuconfig错误: /bin/sh: 1: flex: not found 和 /bin/sh: 1: bison: not found_Imagine Miracle_wxn的博客-CSDN博客https://blog.csdn.net/qq_36393978/article/details/118225531

liuwenjie@liuwenjie-virtual-machine:~/桌面/zuoye3$ sudo make 
liuwenjie@liuwenjie-virtual-machine:~/桌面/zuoye3$ sudo insmod dram.ko
liuwenjie@liuwenjie-virtual-machine:~/桌面/zuoye3$ sudo mknod /dev/dram c 85 0
liuwenjie@liuwenjie-virtual-machine:~/桌面/zuoye3$ gcc -S fileview.cpp  -o fileview.s
liuwenjie@liuwenjie-virtual-machine:~/桌面/zuoye3$ gcc -c fileview.s -o fileview.o
liuwenjie@liuwenjie-virtual-machine:~/桌面/zuoye3$ gcc fileview.cpp -o fileview.out
liuwenjie@liuwenjie-virtual-machine:~/桌面/zuoye3$ ./fileview.out 
liuwenjie@liuwenjie-virtual-machine:~/桌面/zuoye3$ ./fileview.out /dev/dram

?

paging_lowmem.c 源文件

#include <linux/init.h>
#include <linux/module.h>
#include <linux/mm.h>
#include <linux/mm_types.h>
#include <linux/sched.h>
#include <linux/export.h>
/*
该内核模块的主要功能是在内核中申请一个页面,然后利用内核提供的函数按照寻页的步骤一步步查询各级页目录 
最终找到对应的物理地址,这些步骤就像我们手动模拟了MMU单元的寻页过程。 
*/ 

static unsigned long cr0,cr3;

static unsigned long vaddr = 0;

// get_pgtable_macro这个函数的作用是打印页机制中一些重要参数的 
static void get_pgtable_macro(void)
{
	cr0 = read_cr0();
	cr3 = read_cr3_pa();//cr3寄存器的值通过 read_cr3_pa函数来获得 

	printk("cr0 = 0x%lx,cr3 = 0x%lx\n",cr0,cr3);
//_shift这些宏是用来指示线性地址中相应字段所能映射区域大小的对数 
	printk("PGDIR_SHIFT = %d\n",PGDIR_SHIFT);//页全局目录 
	printk("P4D_SHIFT = %d\n",P4D_SHIFT);//4.15内核中新加入的页目录p4d 
    printk("PUD_SHIFT = %d\n",PUD_SHIFT);//页上级目录 
    printk("PMD_SHIFT = %d\n",PMD_SHIFT);//页中级目录 
    printk("PAGE_SHIFT = %d\n",PAGE_SHIFT);//PAGE_SHIFT就是指示page offset字段所能映射区域大小的对数  page offset字段映射的是一个页面的大小 页面大小是4KB,转换为以2为底的对数就是12
	 

//下面这个宏 PTRS_PER_x 是用来指示相应的页目录表中项的个数 
	printk("PTRS_PER_PGD = %d\n",PTRS_PER_PGD);//页全局目录表中项的个数 
    printk("PTRS_PER_P4D = %d\n",PTRS_PER_P4D);//p4d目录表中项的个数 
    printk("PTRS_PER_PUD = %d\n",PTRS_PER_PUD);//页上级目录表中项的个数 
    printk("PTRS_PER_PMD = %d\n",PTRS_PER_PMD);//页中级目录表中项的个数 
	printk("PTRS_PER_PTE = %d\n",PTRS_PER_PTE);// 页表中项的个数 
	printk("PAGE_MASK = 0x%lx\n",PAGE_MASK);// PAGE_MASK是页内偏移掩码 是用来屏蔽掉page offset字段的 
//这些宏是为了方便寻页时进行位运算的 
}
// vaddr2paddr是我们用来进行线性地址到物理地址转换的函数 
static unsigned long vaddr2paddr(unsigned long vaddr)
{
	//为每一个页目录项创建一个变量来将他们保存起来 
	pgd_t *pgd;
	p4d_t *p4d;
	pud_t *pud;
	pmd_t *pmd;
	pte_t *pte;
	unsigned long paddr = 0;
	unsigned long page_addr = 0;
	unsigned long page_offset = 0;
	// pgd_offset函数传入的第一个参数时当前进程的mm_struct结构 
	//我们申请的线性地址是在内核空间的,所以我们要查的页表也是内核页表 
	//所有的进程都共享同一个内核页表,所以可以用当前进程的 
	//我们在查找得到第一个页全局目录项PGD之后,将其作为下级查找的参数传入到p4d_offset中 作下一级的查找 就找到了p4d 
	pgd = pgd_offset(current->mm,vaddr);
	printk("pgd_val = 0x%lx,pgd_index = %lu\n",pgd_val(*pgd),pgd_index(vaddr));
	if(pgd_none(*pgd))
	{
		printk("not mapped in pgd\n");
		return -1;
	}

	p4d = p4d_offset(pgd,vaddr);
	printk("p4d_val = 0x%lx,p4d_index = %lu\n",p4d_val(*p4d),p4d_index(vaddr));
	if(p4d_none(*p4d))
	{
		printk("not mapped in p4d\n");
		return -1;
	}
	//然后再去找相应的PUD页目录项 PMD页目录项 
	pud = pud_offset(p4d,vaddr);
	printk("pud_val = 0x%lx,pud_index = %lu\n",pud_val(*pud),pud_index(vaddr));
	if(pud_none(*pud))
	{
		printk("not mapped in pud\n");
		return -1;
	}

        pmd = pmd_offset(pud,vaddr);
        printk("pmd_val = 0x%lx,pmd_index = %lu\n",pmd_val(*pmd),pmd_index(vaddr));
        if(pmd_none(*pmd))
        {
                printk("not mapped in pmd\n");
                return -1;
        }
	//最后就会找到PTE页表项了 pte_offset_kernel与上面有所不同,这是为了表示我们是在主内核页表中查找的而在进程页表中查找是有一个完全不同的函数 
	//最后我们取得了页表的线性地址 
	pte = pte_offset_kernel(pmd,vaddr);
        printk("pte_val = 0x%lx,pte_index = %lu\n",pte_val(*pte),pte_index(vaddr));
        if(pte_none(*pte))
        {
                printk("not mapped in pte\n");
                return -1;
        }
    //从页表的线性地址也就是这个页表项中取出该页表所映射页框的物理地址 
    //将其与 PAGE_MASK这个变量进行位或操作 取出高48位 就得到了页框的物理地址 
	page_addr = pte_val(*pte) & PAGE_MASK;
	//接下来取出页偏移地址 也就是线性地址中的低12位 
	page_offset = vaddr & ~PAGE_MASK;
	//将两个地址拼接起来 就得到了物理地址 
	paddr = page_addr | page_offset;
	//将这些物理地址都打印出来就得完成线性地址到物理地址转换的过程
	printk("page_addr = %lx,page_offset = %lx\n",page_addr,page_offset);
	printk("vaddr = %lx,paddr = %lx\n",vaddr,paddr);
	return paddr;
}
//内核模块的注册函数 
static int __init v2p_init(void)
{
	unsigned long vaddr = 0;
	printk("vaddr to paddr module is running..\n");
	get_pgtable_macro();
	printk("\n");
	//使用get_free_page函数在内核的ZONE_NORMAL中申请了一块页面 
	//GFP_KERNEL是用来指示它是优先从内存的ZONE_NORMAL区中申请页框的 
	vaddr = __get_free_page(GFP_KERNEL);
	if(vaddr == 0)
	{
		printk("__get_free_page failed..\n");
		return 0;
	}
	sprintf((char *)vaddr,"hello world from kernel");
	printk("get_page_vaddr=0x%lx\n",vaddr);
	vaddr2paddr(vaddr);
	return 0;
}
//内核模块的卸载函数 将我们申请的线性地址空间释放掉 
static void __exit v2p_exit(void)
{
	printk("vaddr to paddr module is leaving..\n");
	free_page(vaddr);
}

module_init(v2p_init);
module_exit(v2p_exit);

MODULE_LICENSE("GPL");

dram.c 源文件

//-------------------------------------------------------------------
//	dram.c
//
//	This module implements a Linux character-mode device-driver
//	for the processor's installed physical memory.  It utilizes
//	the kernel's 'kmap()' function, as a uniform way to provide  
//	access to all the memory-zones (including the "high memory"
//	on systems with more than 896MB of installed physical ram). 
//	The access here is 'read-only' because we deem it too risky 
//	to the stable functioning of our system to allow every user
//	the unrestricted ability to arbitrarily modify memory-areas
//	which might contain some "critical" kernel data-structures.
//	We implement an 'llseek()' method so that users can readily 
//	find out how much physical processor-memory is installed. 
//
//	NOTE: Developed and tested with Linux kernel version 2.6.10
//
//	programmer: ALLAN CRUSE
//	written on: 30 JAN 2005
//	revised on: 28 JAN 2008 -- for Linux kernel version 2.6.22.5
//	revised on: 06 FEB 2008 -- for machines having 4GB of memory
//-------------------------------------------------------------------

#include <linux/module.h>	// for module_init() 
#include <linux/highmem.h>	// for kmap(), kunmap()
#include <linux/uaccess.h>	// for copy_to_user() 

char modname[] = "dram";	// for displaying driver's name
int my_major = 85;		// note static major assignment 
unsigned long dram_size;		// total bytes of system memory

loff_t my_llseek( struct file *file, loff_t offset, int whence );
ssize_t my_read( struct file *file, char *buf, size_t count, loff_t *pos );

struct file_operations 
my_fops =	{
		owner:		THIS_MODULE,
		llseek:		my_llseek,
		read:		my_read,
		};

static int __init dram_init( void )
{
	printk( "<1>\nInstalling \'%s\' module ", modname );
	printk( "(major=%d)\n", my_major );
   
	dram_size = 0x25f5ffff8;
	printk( "<1>  ramtop=%08lX (%lu MB)\n", dram_size, dram_size >> 20 );
	return 	register_chrdev( my_major, modname, &my_fops );
}

static void __exit dram_exit( void )
{
	unregister_chrdev( my_major, modname );
	printk( "<1>Removing \'%s\' module\n", modname );
}

ssize_t my_read( struct file *file, char *buf, size_t count, loff_t *pos )
{
	struct page	*pp;
	void		*from;
	int		page_number, page_indent, more;
	
	// we cannot read beyond the end-of-file
	if ( *pos >= dram_size ) return 0;

	// determine which physical page to temporarily map
	// and how far into that page to begin reading from 
	page_number = *pos / PAGE_SIZE;
	page_indent = *pos % PAGE_SIZE;
	
	// map the designated physical page into kernel space
	/*If kerel vesion is 2.6.32 or later, please use pfn_to_page() to get page, and include
	    asm-generic/memory_model.h*/

       pp = pfn_to_page( page_number);
	
	from = kmap( pp ) + page_indent;
	
	// cannot reliably read beyond the end of this mapped page
	if ( page_indent + count > PAGE_SIZE ) count = PAGE_SIZE - page_indent;

	// now transfer count bytes from mapped page to user-supplied buffer 	
	more = copy_to_user( buf, from, count );
	
	// ok now to discard the temporary page mapping
	kunmap( pp );
	
	// an error occurred if less than count bytes got copied
	if ( more ) return -EFAULT;
	
	// otherwise advance file-pointer and report number of bytes read
	*pos += count;
	return	count;
}

loff_t my_llseek( struct file *file, loff_t offset, int whence )
{
	loff_t	newpos = -1;

	switch( whence )
		{
		case 0: newpos = offset; break;			// SEEK_SET
		case 1: newpos = file->f_pos + offset; break; 	// SEEK_CUR
		case 2: newpos = dram_size + offset; break; 	// SEEK_END
		}

	if (( newpos < 0 )||( newpos > dram_size )) return -EINVAL;
	file->f_pos = newpos;
	return	newpos;
}

MODULE_LICENSE("GPL");
module_init( dram_init );
module_exit( dram_exit );

fileview.cpp 源文件

//----------------------------------------------------------------
//	fileview.cpp
//
//	This program displays the contents of a specified file 
//	in hexadecimal and ascii formats (including any device
//	special files representing storage media).  A user may
//	navigate the file's contents using arrow-key commands,
//	or may adjust the format of the hexadecimal display to
//	select from among five data-sizes: byte (B), word (W), 
//	doubleword (D), quadword (Q) or octaword (O).  It also
//	is possible to seek to a specified position within the
//	file by hitting the <ENTER>-key and then typing in the
//	desired (hexadecimal) address.  Type <ESCAPE> to quit.  
//
//	       compile-and-link using: $ make fileview
//
//	programmer: ALLAN CRUSE
//	written on: 26 OCT 2002
//	revised on: 07 JUN 2006 -- removed reliance on 'ncurses' 
//----------------------------------------------------------------

#include <stdio.h>	// for printf(), perror(), fflush() 
#include <fcntl.h>	// for open()
#include <string.h>	// for strncpy()
#include <unistd.h>	// for read(), lseek64()
#include <stdlib.h>	// for exit()
#include <termios.h>	// for tcgetattr(), tcsetattr()

#define MAXNAME	80
#define BUFHIGH 16
#define BUFWIDE 16
#define BUFSIZE 256
#define ROW	6
#define COL	2

#define KB_SEEK 0x0000000A
#define KB_QUIT	0x0000001B
#define KB_BACK 0x0000007F
#define KB_HOME	0x00315B1B
#define KB_LNUP 0x00415B1B
#define KB_PGUP	0x00355B1B
#define KB_LEFT 0x00445B1B
#define KB_RGHT 0x00435B1B
#define KB_LNDN 0x00425B1B
#define KB_PGDN 0x00365B1B
#define KB_END  0x00345B1B
#define KB_DEL  0x00335B1B


char progname[] = "FILEVIEW";
char filename[ MAXNAME + 1 ];
char buffer[ BUFSIZE ];
char outline[ 80 ];

int main( int argc, char *argv[] )
{
	// setup the filename (if supplied), else terminate
	if ( argc > 1 ) strncpy( filename, argv[1], MAXNAME );
 	else { fprintf( stderr, "argument needed\n" ); exit(1); }

	// open the file for reading
	int	fd = open( filename, O_RDONLY );
	if ( fd < 0 ) { perror( filename ); exit(1); }

	// obtain the filesize (if possible)
	long long	filesize = lseek64( fd, 0LL, SEEK_END );
	if ( filesize < 0LL ) 
		{ 
		fprintf( stderr, "cannot locate \'end-of-file\' \n" ); 
		exit(1); 
		}

	long long	incmin = ( 1LL <<  8 );
	long long	incmax = ( 1LL << 36 );		
	long long	posmin = 0LL;
	long long	posmax = (filesize - 241LL)&~0xF;
	if ( posmax < posmin ) posmax = posmin;

	// initiate noncanonical terminal input
	struct termios	tty_orig;
	tcgetattr( STDIN_FILENO, &tty_orig );
	struct termios	tty_work = tty_orig;
	tty_work.c_lflag &= ~( ECHO | ICANON );  // | ISIG );
	tty_work.c_cc[ VMIN ]  = 1;
	tty_work.c_cc[ VTIME ] = 0;
	tcsetattr( STDIN_FILENO, TCSAFLUSH, &tty_work );	
	printf( "\e[H\e[J" );

	// display the legend
	int	i, j, k;
	k = (77 - strlen( progname ))/2;
	printf( "\e[%d;%dH %s ", 1, k, progname );
	k = (77 - strlen( filename ))/2;
	printf( "\e[%d;%dH\'%s\'", 3, k, filename );
	char	infomsg[ 80 ];
	sprintf( infomsg, "filesize: %llu (=0x%013llX)", filesize, filesize );
	k = (78 - strlen( infomsg ));
	printf( "\e[%d;%dH%s", 24, k, infomsg );
	fflush( stdout );

	// main loop to navigate the file
	long long	pageincr = incmin;
	long long	lineincr = 16LL;
	long long	position = 0LL;
	long long	location = 0LL;
	int		format = 1;
	int		done = 0;
	while ( !done )
		{
		// erase prior buffer contents
		for (j = 0; j < BUFSIZE; j++) buffer[ j ] = ~0;

		// restore 'pageincr' to prescribed bounds
		if ( pageincr == 0LL ) pageincr = incmax;
		else if ( pageincr < incmin ) pageincr = incmin;
		else if ( pageincr > incmax ) pageincr = incmax;
		
		// get current location of file-pointer position
		location = lseek64( fd, position, SEEK_SET );
		
		// try to fill 'buffer[]' with data from the file
		char	*where = buffer;
		int	to_read = BUFSIZE;
		while ( to_read > 0 )
			{
			int	nbytes = read( fd, where, to_read );
			if ( nbytes <= 0 ) break; 
			to_read -= nbytes;
			where += nbytes;
			}
		int	datalen = BUFSIZE - to_read; 

		// display the data just read into the 'buffer[]' array
		unsigned char		*bp;
		unsigned short		*wp;
		unsigned int		*dp;
		unsigned long long	*qp;
		for (i = 0; i < BUFHIGH; i++)
			{
			int	linelen;

			// draw the line-location (13-digit hexadecimal)
			linelen = sprintf( outline, "%013llX ", location );
			
			// draw the line in the selected hexadecimal format
			switch ( format )
				{
				case 1:	// 'byte' format
				bp = (unsigned char*)&buffer[ i*BUFWIDE ];
				for (j = 0; j < BUFWIDE; j++)
					linelen += sprintf( outline+linelen, 
						"%02X ", bp[j] );
				break;

				case 2:	// 'word' format
				wp = (unsigned short*)&buffer[ i*BUFWIDE ];
				for (j = 0; j < BUFWIDE/2; j++)
					linelen += sprintf( outline+linelen,
						" %04X ", wp[j] );
				break;

				case 4:	// 'dword' format
				dp = (unsigned int*)&buffer[ i*BUFWIDE ];
				for (j = 0; j < BUFWIDE/4; j++)
					linelen += sprintf( outline+linelen,
						"  %08X  ", dp[j] );
				break;

				case 8:	// 'qword' format
				qp = (unsigned long long*)&buffer[ i*BUFWIDE ];
				for (j = 0; j < BUFWIDE/8; j++)
					linelen += sprintf( outline+linelen,
						"    %016llX    ", qp[j] );
				break;

				case 16: // 'octaword'
				qp = (unsigned long long*)&buffer[ i*BUFWIDE ];
				linelen += sprintf( outline+linelen, "     " );
				linelen += sprintf( outline+linelen, 
					"   %016llX%016llX   ", qp[1], qp[0] );
				linelen += sprintf( outline+linelen, "     " ); 
				break;
				}

			// draw the line in ascii format
			for (j = 0; j < BUFWIDE; j++)
				{
				char	ch = buffer[ i*BUFWIDE + j ];
				if (( ch < 0x20 )||( ch > 0x7E )) ch = '.';
				linelen += sprintf( outline+linelen, "%c", ch);
				}

			// transfer this output-line to the screen 
			printf( "\e[%d;%dH%s", ROW+i, COL, outline );

			// advance 'location' for the next output-line
			location += BUFWIDE;
			} 	
		printf( "\e[%d;%dH", 23, COL );
		fflush( stdout );	
	
		// await keypress 	
		long long	inch = 0LL;
		read( STDIN_FILENO, &inch, sizeof( inch ) );
		printf( "\e[%d;%dH%60s", 23, COL, " " );	

		// interpret navigation or formatting command
		inch &= 0x00FFFFFFLL;
		switch ( inch )
			{
			// move to the file's beginning/ending
			case 'H': case 'h':
			case KB_HOME:	position = posmin; break;
			case 'E': case 'e':
			case KB_END:	position = posmax; break;

			// move forward/backward by one line
			case KB_LNDN:	position += BUFWIDE; break;
			case KB_LNUP:	position -= BUFWIDE; break;

			// move forward/packward by one page
			case KB_PGDN:	position += pageincr; break;
			case KB_PGUP:	position -= pageincr; break;

			// increase/decrease the page-size increment
			case KB_RGHT:	pageincr >>= 4; break;
			case KB_LEFT:	pageincr <<= 4; break;

			// reset the hexadecimal output-format
			case 'B': case 'b':	format = 1; break;
			case 'W': case 'w':	format = 2; break;
			case 'D': case 'd':	format = 4; break;
			case 'Q': case 'q':	format = 8; break;
			case 'O': case 'o':	format = 16; break;

			// seek to a user-specified file-position
			case KB_SEEK:
			printf( "\e[%d;%dHAddress: ", 23, COL );
			fflush( stdout );
			{
			char	inbuf[ 16 ] = {0};
				//tcsetattr( STDIN_FILENO, TCSANOW, &tty_orig );
			int	i = 0;
			while ( i < 15 )
				{
				char ch = 0;
				read( STDIN_FILENO, &ch, sizeof( ch ) );
				ch &= 0xFFFFFF;
				if ( ch == '\n' ) break;
				if ( ch == KB_QUIT ) { inbuf[0] = 0; break; }
				if ( ch == KB_LEFT ) ch = KB_BACK;
				if ( ch == KB_DEL ) ch = KB_BACK;
				if (( ch == KB_BACK )&&( i > 0 ))
					{ 
					inbuf[--i] = 0; 
					printf( "\b \b" ); 
					fflush( stdout );
					}
				if (( ch < 0x20 )||( ch > 0x7E )) continue;
				inbuf[ i++ ] = ch;
				printf( "%c", ch );
				fflush( stdout );
				}		
			printf( "\e[%d;%dH%70s", 23, COL, " " );
			fflush( stdout );
			position = strtoull( inbuf, NULL, 16 );
			position &= ~0xFLL;	// paragraph align
			}
			break;			

			// program termination 
			case KB_QUIT:	done = 1; break;

			default:	
			printf( "\e[%d;%dHHit <ESC> to quit", 23, 2 ); 
			}
		fflush( stdout );

		// insure that 'position' remains within bounds
		if ( position < posmin ) position = posmin;
		if ( position > posmax ) position = posmax;
		}	

	// restore canonical terminal behavior
	tcsetattr( STDIN_FILENO, TCSAFLUSH, &tty_orig );	
	printf( "\e[%d;%dH\e[0J\n", 23, 0 );
}

  系统运维 最新文章
配置小型公司网络WLAN基本业务(AC通过三层
如何在交付运维过程中建立风险底线意识,提
快速传输大文件,怎么通过网络传大文件给对
从游戏服务端角度分析移动同步(状态同步)
MySQL使用MyCat实现分库分表
如何用DWDM射频光纤技术实现200公里外的站点
国内顺畅下载k8s.gcr.io的镜像
自动化测试appium
ctfshow ssrf
Linux操作系统学习之实用指令(Centos7/8均
上一篇文章      下一篇文章      查看所有文章
加:2021-10-04 13:11:34  更:2021-10-04 13:13:04 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/4 18:35:31-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码