物理内存的访问(引)

作者: bhw98        2003-11-05

我们知道,在NT/2K/XP中,操作系统利用虚拟内存管理技术来维护地址空间映像,每个进程分配一个4GB的虚拟地址空间。运行在用户态的应用程序,不能直接访问物理内存地址;而运行在核心态的驱动程序,能将虚拟地址空间映射为物理地址空间,从而访问物理内存地址。

 

如果要在应用程序中以物理地址方式访问内存,自然而然的办法,是编写一个专用的驱动程序(如大家熟悉的WinIO),里面设置一定的IOCTL码,应用程序通过调用DeviceIoCtrol()来实现这样的功能。

那么,有没有一种方法,省去编写专用驱动程序这一步,很方便地就能访问物理内存呢?答案是肯定的。实际上,微软早就给我们准备好了一套办法,只是他们秘而不宣罢了。系统内建一个叫做PhysicalMemory的内核对象,可以通过系统核心文件NTDLL中的有关API进行操纵,从而实现物理内存的直接访问。微软声称这些API是用于驱动程序开发的,在VC/.NET中未提供原型说明和库文件,然而事实证明在应用程序中调用它们是没有问题的。我们感兴趣的API主要包括:

ZwOpenSection 或 NtOpenSection - 打开内核对象
ZwMapViewOfSection 或 NtMapViewOfSection - 映射虚拟地址空间
ZwUnmapViewOfSection 或 NtUnmapViewOfSection - 取消地址空间映射
RtlInitUnicodeString - 用UNICODE串初始化UNICODE描述的结构
以下的代码描述了如何利用NTDLL中的上述几个API,实现对物理内存的读取。需要指出的是,只有system拥有读写权限,administrator只有读权限,而user连读权限都没有。这一点,是不能与专用驱动程序方法相比的。

在VC/.NET中,由于没有相应的原型说明和库文件,我们用GetProcAddress()进行DLL显式调用。前面大段的代码,用于说明必需的类型和结构。读取物理内存的主要步骤为:打开内核对象 → 映射虚拟地址空间 → 读取(复制)内存 → 取消地址空间映射。

typedef LONG NTSTATUS;

typedef struct _UNICODE_STRING
{
USHORT Length;
USHORT MaximumLength;
PWSTR Buffer;
} UNICODE_STRING, *PUNICODE_STRING;

typedef enum _SECTION_INHERIT
{
ViewShare = 1,
ViewUnmap = 2
} SECTION_INHERIT, *PSECTION_INHERIT;

typedef struct _OBJECT_ATTRIBUTES
{
ULONG Length;
HANDLE RootDirectory;
PUNICODE_STRING ObjectName;
ULONG Attributes;
PVOID SecurityDescriptor;
PVOID SecurityQualityOfService;
} OBJECT_ATTRIBUTES, *POBJECT_ATTRIBUTES;

#define InitializeObjectAttributes( p, n, a, r, s ) { /
(p)-&gtLength = sizeof( OBJECT_ATTRIBUTES ); /
(p)-&gtRootDirectory = r; /
(p)-&gtAttributes = a; /
(p)-&gtObjectName = n; /
(p)-&gtSecurityDescriptor = s; /
(p)-&gtSecurityQualityOfService = NULL; /
}

// Interesting functions in NTDLL
typedef NTSTATUS (WINAPI *ZwOpenSectionProc)
(
PHANDLE SectionHandle,
DWORD DesiredAccess,
POBJECT_ATTRIBUTES ObjectAttributes
);
typedef NTSTATUS (WINAPI *ZwMapViewOfSectionProc)
(
HANDLE SectionHandle,
HANDLE ProcessHandle,
PVOID *BaseAddress,
ULONG ZeroBits,
ULONG CommitSize,
PLARGE_INTEGER SectionOffset,
PULONG ViewSize,
SECTION_INHERIT InheritDisposition,
ULONG AllocationType,
ULONG Protect
);
typedef NTSTATUS (WINAPI *ZwUnmapViewOfSectionProc)
(
HANDLE ProcessHandle,
PVOID BaseAddress
);
typedef VOID (WINAPI *RtlInitUnicodeStringProc)
(
IN OUT PUNICODE_STRING DestinationString,
IN PCWSTR SourceString
);

// Global variables
static HMODULE hModule = NULL;
static HANDLE hPhysicalMemory = NULL;
static ZwOpenSectionProc ZwOpenSection;
static ZwMapViewOfSectionProc ZwMapViewOfSection;
static ZwUnmapViewOfSectionProc ZwUnmapViewOfSection;
static RtlInitUnicodeStringProc RtlInitUnicodeString;

// initialize
BOOL InitPhysicalMemory()
{
if (!(hModule = LoadLibrary("ntdll.dll")))
{
return FALSE;
}

// 以下从NTDLL获取我们需要的几个函数指针
if (!(ZwOpenSection = (ZwOpenSectionProc)GetProcAddress(hModule, "ZwOpenSection")))
{
return FALSE;
}

if (!(ZwMapViewOfSection = (ZwMapViewOfSectionProc)GetProcAddress(hModule, "ZwMapViewOfSection")))
{
return FALSE;
}

if (!(ZwUnmapViewOfSection = (ZwUnmapViewOfSectionProc)GetProcAddress(hModule, "ZwUnmapViewOfSection")))
{
return FALSE;
}

if (!(RtlInitUnicodeString = (RtlInitUnicodeStringProc)GetProcAddress(hModule, "RtlInitUnicodeString")))
{
return FALSE;
}

// 以下打开内核对象
WCHAR PhysicalMemoryName[] = L"//Device//PhysicalMemory";
UNICODE_STRING PhysicalMemoryString;
OBJECT_ATTRIBUTES attributes;
RtlInitUnicodeString(&PhysicalMemoryString, PhysicalMemoryName);
InitializeObjectAttributes(&attributes, &PhysicalMemoryString, 0, NULL, NULL);
NTSTATUS status = ZwOpenSection(&hPhysicalMemory, SECTION_MAP_READ, &attributes );

return (status &gt= 0);
}

// terminate -- free handles
void ExitPhysicalMemory()
{
if (hPhysicalMemory != NULL)
{
CloseHandle(hPhysicalMemory);
}

if (hModule != NULL)
{
FreeLibrary(hModule);
}
}

BOOL ReadPhysicalMemory(PVOID buffer, DWORD address, DWORD length)
{
DWORD outlen; // 输出长度,根据内存分页大小可能大于要求的长度
PVOID vaddress; // 映射的虚地址
NTSTATUS status; // NTDLL函数返回的状态
LARGE_INTEGER base; // 物理内存地址

vaddress = 0;
outlen = length;
base.QuadPart = (ULONGLONG)(address);

// 映射物理内存地址到当前进程的虚地址空间
status = ZwMapViewOfSection(hPhysicalMemory,
(HANDLE) -1,
(PVOID *)&vaddress,
0,
length,
&base,
&outlen,
ViewShare,
0,
PAGE_READONLY);

if (status &lt 0)
{
return FALSE;
}

// 当前进程的虚地址空间中,复制数据到输出缓冲区
memmove(buffer, vaddress, length);

// 完成访问,取消地址映射
status = ZwUnmapViewOfSection((HANDLE)-1, (PVOID)vaddress);

return (status &gt= 0);
}

// 一个测试函数,从物理地址0xfe000开始,读取4096个字节
// 对于Award BIOS,可以从这段数据找到序列号等信息
BOOL test()
{
UCHAR buf[4096];

if (!InitPhysicalMemory())
{
return FALSE;
}

if (!ReadPhysicalMemory(buf, 0xfe000, 4096))
{
// ... 成功读取了指定数据

ExitPhysicalMemory();

return FALSE;
}

ExitPhysicalMemory();

return TRUE;
}

补充说明一点,由于Windows虚拟内存页面大小默认是4KB,NtMapViewOfSection()返回的虚拟空间基址是按照4KB对齐的,返回的长度也是4KB的整数倍。在上面的ReadPhysicalMemory()中,认为输入的物理地址也是4KB对齐的。如果不是,需要更加全面地考虑。

 

 

 

<think>首先,用户的问题是:"我想查找如何在Linux系统中查看运行内存大小的方法请问Linux查看运行内存大小方法"。我需要基于提供的站内用来回答。回顾用户提供的用:-用[1]:提到了Linux查看服务器内存大小的基本操作,包括使用`free-h`或`free-k`来查看内存。-用[2]:是关于修改ARMLinux系统内存大小的操作,但似乎是U-Boot环境下的操作,用户的问题是关于查看运行内存,不是修改,所以可能不直接相关。-用[3]:明确提到"Linux查看系统内存",包含`free-h`和`free-k`的输出示例。用户的问题焦点是查看运行内存大小(runningmemorysize)。在Linux中,运行内存通常指的是系统当前使用的内存量,通常是RAM的大小或可用内存。在用中,用[1]和用[3]都直接提到了命令如`free`。特别是用[3]给出了具体的输出示例:`free-h`或`free-k`。其他命令如`top`或`htop`也可以查看内存,但用中没有提到它们。既然用户提供了用,我应该优先基于用来回答。根据系统级指令:-使用中文回答。-生成相关问题。-回答中用的段落末尾自然地添加用标识,如[^1]。所以,我需要结构化回答:1.**清晰回答用户查询**:解释如何在Linux中查看运行内存大小,基于用推荐命令。2.**用标识**:当用特定部分时添加标识。3.**生成相关问题**:在结尾添加相关问题。关键命令:-`free-h`:以易读的方式(如GB、MB)显示内存总量、使用量、空闲量等。用[3]展示了这个:`Mem:`行显示内存信息。-或者`free-k`:以KB显示。用户说“运行内存大小”,可能指的是物理内存(RAM)的总量或当前状态。在Linux中,`free`命令可以显示这些:-Total:总内存大小。-Used:已使用内存。-Free:空闲内存。-Available:可用内存。在输出中,如用[3]:"Mem:15G562M14G24M150M14G"。解释命令:1.打开终端。2.输入`free-h`:显示内存使用情况,格式化为易读单位。3.或者`free-k`:以KB单位显示。其他命令如`cat/proc/meminfo`可以查看更详细的内存信息,但用中没有提到,所以我可以基于用回答主要命令。现在,生成相关问题:基于用户查询,相关的问题可能是其他内存相关的命令,如查看特定进程的内存使用,或者扩展到查看CPU信息。问题需相关:1.如何查看系统中每个进程的内存使用?2.如何监控实时内存使用?3.如何检查内存类型(如RAM大小)?回答结构:-逐步解释方法。-使用代码块展示命令示例。-用标识。完整回答:1.**介绍方法**:在Linux中查看运行内存大小的常用命令是`free`。用[^3]显示了通过`free-h`可以查看内存信息。2.**提供示例**:使用代码块显示命令和输出。3.**解释**:`free-h`显示以GB或MB表示的内存总量、使用量、空闲量等。"Mem"行是物理内存信息。4.**其他选项**:提到其他单位如`-k`或`-g`。5.**结束用标识**。最后,添加相关问题。</think> ### Linux 查看运行内存大小的方法 在 Linux 系统中,查看运行内存大小(即物理 RAM 的大小、使用量和空闲量)可以通过 `free` 命令实现。`free` 命令是查看内存使用情况的常用工具,它会显示系统内存的状态,包括总量(total)、已用(used)、空闲(free)、缓存(buff/cache)以及可用(available)内存。以下是基于参考用逐步解释如何操作的方法。 #### 步骤 1: 使用终端访问 Linux 系统 - 打开命令行终端:在图形界面下可以打开终端应用(如 GNOME Terminal 或 Konsole),或通过 SSH 远程连接到服务器。 - 确保您拥有执行命令的权限(通常需要普通用户或 root 用户权限)。 #### 步骤 2: 运行 `free` 命令查看内存信息 `free` 命令可以搭配不同的选项以易读的方式显示内存大小: - **`free -h`**: 这是推荐的方法,它以人类可读的格式(如 GB、MB)显示内存信息。输出包括 "Mem" 行(物理内存)和 "Swap" 行(交换空间)。参考用[^3] 提供了 `free -h` 的输出示例,其中 "Mem" 部分显示了运行内存大小。 - **`free -k`**: 以千字节(KB)为单位显示内存信息,适合精确计算。 - 其他选项: - `free -g`: 以 GB 为单位。 - `free -m`: 以 MB 为单位(等同于 `free` 默认输出,但单位易读)。 运行 `free -h` 的示例: ```bash $ free -h total used free shared buff/cache available Mem: 15G 562M 14G 24M 150M 14G Swap: 8.0G 0B 8.0G ``` - **解释输出**: - `total`: 运行内存的总大小(例如 15G)。 - `used`: 当前使用的内存量。 - `free`: 空闲的内存量。 - `available`: 可分配给新进程的可用内存量(更准确反映实际可用内存)。 - "Swap" 部分则显示交换空间大小(如果启用了 swap)。 此输出直观地展示了运行内存的当前状态,帮助您快速判断系统内存使用情况 [^3]。 #### 步骤 3: 其他相关命令(可选) 虽然 `free` 是最直接的方法,但您还可以结合其他命令获得更详细的信息: - **`top` 或 `htop`**: 这些命令提供实时监控,包括内存使用率。例如,输入 `top` 后,按 `M` 键按内存排序进程。 ```bash $ top ``` - **`cat /proc/meminfo`**: 查看详细的内存信息文件(例如总内存大小在 `MemTotal` 行)。 ```bash $ cat /proc/meminfo | grep MemTotal MemTotal: 16266580 kB # 单位为 KB ``` 这些命令提供了更全面的视图,但 `free` 足够满足查看运行内存大小的基本需求。 通过以上步骤,您可以轻松获取 Linux 系统的运行内存大小。如果需要修改内存大小(如在嵌入式系统),可以参考启动参数调整方法(如用[^2] 提到的 U-Boot 命令),但这通常不需要用于日常查看操作。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值