当前位置:文档之家› 操作系统安全实验及代码

操作系统安全实验及代码

操作系统安全实验及代码
操作系统安全实验及代码

操作系统系统安全实验

实验一获取系统信息

1.实验目的:

了解API函数获取心态信息方法熟悉VC程序设计

2.实验内容:

利用API函数,获取系统软、硬件信息。主要包括硬件方面:处理机信息(架构、级别、数量、版本),内存信息(当前内存大小、可用内存大小、虚拟内存大小),磁盘信息(磁盘卷标、序列号、总空间、可用空间等);软件方面:操作系统信息(主版本号、次版本号、补丁版本号等),文件信息(文件系统类型、操作系统目录、可执行文件目录)。

3.参见参考代码。

①getmemInf.cpp

#include

#include

using namespace std;

int main(int argc, char** argv)

{

MEMORYSTATUS mem;

ZeroMemory(&mem, sizeof(mem));

GlobalMemoryStatus(&mem);

mem.dwLength = sizeof(mem);

cout << "物理内存:" << mem.dwTotalPhys /1024 /1024 << endl;

cout << "剩余内存:" << mem.dwAvailPhys / 1024 /1024 << endl;

cout << "虚拟内存:" << mem.dwTotalVirtual /1024 / 1024 << endl;

cout << "剩余虚拟内存:" << mem.dwAvailVirtual / 1024 / 1024 << endl;

system("PAUSE");

return 0;

}

②getOSinf.cpp

#include

#include

#pragma warning(disable:4996)

using namespace std;

int main(int argc, char** argv)

{

OSVERSIONINFOEX os;

bool ret;

os.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);

ret = GetVersionEx((OSVERSIONINFO *)&os); // Not OSVERSIONINFOEX

if (ret == false)

{

cout << "GetVersionEx error." << endl;

return -1;

}

cout << "主版本号:" << os.dwMajorVersion << endl;

cout << "次版本号:" << os.dwMinorVersion << endl;

cout << "编译次数:" << os.dwBuildNumber << endl;

cout << "平台类型:" << os.dwPlatformId << endl; //ver_platform_* cout << "产品类型:" << os.wProductType << endl; //ver_nt_*

cout << "程序组掩码" << os.wSuiteMask << endl; //ver_suite_*

cout << "补丁主版本:" << os.wServicePackMajor << endl;

cout << "补丁次版本:" << os.wServicePackMinor << endl;

system("PAUSE");

}

③getSysinf.cpp

#include

#include

#include

using namespace std;

typedef struct MyVolumnInfo

{

TCHAR diskRootPath[256];

TCHAR volNameBuf[256];

DWORD volSize;

DWORD volSerialNumber;

DWORD maxComponetLength;

DWORD fileSystemFlag;

TCHAR fileSystemName[256];

DWORD fileSystemNameSize;

}MYVOLUMNINFO;

int main(int argc, char** argv)

{

TCHAR winDir[256];

TCHAR sysDir[256];

TCHAR diskName[256]; //不可小于256

TCHAR diskLogicName[256];

DWORD diskString;

LPWSTR szDiskName;

MYVOLUMNINFO volInfo;

UINT64 diskAvail, diskTotal, diskFree;

SYSTEM_INFO sysInf;

ZeroMemory(winDir, sizeof(winDir));

ZeroMemory(sysDir, sizeof(sysDir));

ZeroMemory(diskName, sizeof(diskName));

ZeroMemory(&volInfo, sizeof(volInfo));

ZeroMemory(diskLogicName, sizeof(diskLogicName));

ZeroMemory(&sysInf, sizeof(sysInf));

GetSystemInfo(&sysInf);

GetWindowsDirectory(winDir, sizeof(winDir));

GetSystemDirectory(sysDir, sizeof(sysDir));

wcout << "处理器架构:" << sysInf.wProcessorArchitecture << endl;

wcout << "处理器类型:" << sysInf.dwProcessorType << endl;

wcout << "处理器数量:" << sysInf.dwNumberOfProcessors << endl;

wcout << "处理器级别:" << sysInf.wProcessorLevel << endl;

wcout << "处理器版本:" << sysInf.wProcessorRevision << endl;

wcout << "处理器掩码:" << sysInf.dwActiveProcessorMask << endl; //3=11B,表示两个都用

wcout << "页面大小:" << sysInf.dwPageSize << endl;

wcout << "虚拟内存颗粒:" << sysInf.dwAllocationGranularity << endl;

wcout << "应用程序最小地址:" << sysInf.lpMinimumApplicationAddress << endl;

wcout << "应用程序最大地址:" << sysInf.lpMaximumApplicationAddress << endl;

wcout << "Windows:" << winDir << endl;

wcout << "System:" << sysDir << endl;

diskString = GetLogicalDrives();

cout << bitset<32> (diskString) << endl;

GetLogicalDriveStrings(sizeof(diskLogicName), diskLogicName);

szDiskName = (LPWSTR)diskLogicName;

while (*szDiskName != '\0\0')

{

wcout << szDiskName << endl;

GetVolumeInformation(szDiskName, volInfo.volNameBuf, 255, &volInfo.volSerialNumber, \ &volInfo.maxComponetLength, &volInfo.fileSystemFlag, volInfo.fileSystemName, 255);

wcout << "卷标名称:" << volInfo.volNameBuf << endl;

wcout << "磁盘序列号:" << volInfo.volSerialNumber << endl;

wcout << "最大文件名长度:" << volInfo.maxComponetLength << endl;

wcout << "文件系统标志:" << volInfo.fileSystemFlag << endl;

wcout << "文件系统名称:" << volInfo.fileSystemName << endl;

GetDiskFreeSpaceEx(szDiskName, (PULARGE_INTEGER)&diskAvail, (PULARGE_INTEGER)&diskTotal, \

(PULARGE_INTEGER)&diskFree);

wcout << "可用空间:" << diskAvail / 1024 / 1024 << "MB" << endl;

wcout << "总空间:" << diskTotal / 1024 / 1024 << "MB" << endl;

wcout << "剩余空间:" << diskFree / 1024 / 1024 << "MB" << endl;

szDiskName = szDiskName + lstrlen(szDiskName) + 1;

ZeroMemory(&volInfo, sizeof(volInfo));

}

system("PAUSE");

return 0;

}

实验二系统注册表操作

实验目的:

了解注册表基本常识

了解注册表项目含义

利用API函数添加、删除、修改注册表

实验内容

1. 编写代码,打开指定注册表项目,获得项目取值。

2. 编写代码,修改注册表项目取值。

3. 编写代码,为指定程序添加自启动功能。

实验步骤:

参见代码。

regedit.cpp

#include

#include

using namespace std;

int main(int argc, char** argv)

{

HKEY keyRoot;

HKEY keyCur;

TCHAR csPath[512] = { 0 };

LONG lRet;

TCHAR tsValue[128] = { 0 };

TCHAR tsGetValue[128] = { 0 };

DWORD dwValueSize;

lRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, NULL, 0, KEY_ALL_ACCESS, &keyCur);

if (lRet != ERROR_SUCCESS)

{

wcout << TEXT("RegOpenKeyEx error.") << endl;

wcout << GetLastError() << endl;

system("PAUSE");

exit(0);

}

wcout << keyCur << endl;

lRet = RegCreateKey(keyCur, //创建的起始键值

TEXT("software\\AddSubKey"), //路径表示的添加子键名

&keyCur);

if (lRet != ERROR_SUCCESS)

{

wcout << TEXT("RegCreateKeyEx error.") << endl;

wcout << GetLastError() << endl;

system("PAUSE");

exit(0);

}

lstrcpy(tsValue, TEXT("D:\\test\\test.txt")); //项目取值lRet = RegSetValueEx(keyCur, //添加项目的键路径TEXT("PATH"), //添加的子键项目

NULL, //保留值

REG_SZ, //数据类型

(LPBYTE)tsValue, //项目取值

sizeof(tsValue)); //取值大小

if (lRet != ERROR_SUCCESS)

{

wcout << TEXT("RegSetValueEx error.") << endl;

wcout << GetLastError() << endl;

system("PAUSE");

exit(0);

}

wcout << keyCur << endl;

system("PAUSE");

lRet = RegGetValue(keyCur,

NULL,

TEXT("PATH"),

RRF_RT_ANY,

NULL,

(LPBYTE)tsGetValue,

&dwValueSize);

if (lRet != ERROR_SUCCESS)

{

wcout << TEXT("RegGetValue error.") << endl;

wcout << GetLastError() << endl;

system("PAUSE");

exit(0);

}

wcout << tsGetValue << endl;

lRet = RegDeleteKeyValue(keyCur, //删除键

NULL, //键扩展路径

TEXT("PATH")); //项目名称

wcout << keyCur << endl;

lRet = RegDeleteKeyEx(HKEY_LOCAL_MACHINE, //Key句柄

TEXT("software\\AddSubKey"),

//NULL, //删除子键

KEY_WOW64_32KEY, //32位系统

0);

if (lRet != ERROR_SUCCESS)

{

wcout << TEXT("RegDelKey error.") << endl;

wcout << GetLastError() << endl;

system("PAUSE");

exit(0);

}

system("PAUSE");

RegCloseKey(keyCur);

return 0;

}

/*VOID GetLastErrorInfo()

{

LPVOID lpMsgError;

TCHAR msg[512] = { 0 };

FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL,

GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR)&lpMsgError, 0, NULL);

wsprintf(msg, TEXT("ERROR Code:%ld:%s"), GetLastError(), lpMsgError);

wcout << msg << endl;

}

int main(int argc, char** argv)

{

TCHAR sKey[64] = { 0 };

TCHAR sValue[512] = { 0 };

DWORD valueSize = sizeof(sValue);

DWORD regType = REG_SZ;

LONG lResult = -1;

HKEY regKey;

lstrcpy(sValue, TEXT("D:\\test\\test"));

lResult = RegOpenKeyEx(HKEY_LOCAL_MACHINE, TEXT("software\\Microsoft\\Windows\\"), 0, KEY_ALL_ACCESS, ®Key);

if (lResult != ERROR_SUCCESS)

{

wcout << "HEHE" << endl;

GetLastErrorInfo();

system("PAUSE");

return -1;

}

//lResult = RegGetValue(regKey, nullptr, NULL, RRF_RT_ANY, NULL, (LPBYTE)&sValue, &valueSize);

//lResult = RegSetValueEx(regKey, TEXT("PATH"), NULL, REG_SZ,(LPBYTE)&sValue, sizeof(sValue));

//GetLastErrorInfo();

//lResult = RegDeleteKeyValue(regKey, NULL, TEXT("PATH"));

lResult = RegDeleteKeyEx(regKey, TEXT("MYKEY"), KEY_WOW64_32KEY, 0);

wcout << lResult << endl;

wcout << sValue << endl;

RegCloseKey(regKey);

system("PAUSE");

return 0;

}

*/

/*RegGetValue*/

/*

lResult = RegOpenKeyEx(HKEY_LOCAL_MACHINE, TEXT("software\\Microsoft\\Windows\\MYKEY\\"), 0, KEY_EXECUTE, ®Key);

if (lResult != ERROR_SUCCESS)

{

wcout << "HEHE" << endl;

GetLastErrorInfo();

system("PAUSE");

return -1;

}

lResult = RegGetValue(regKey, nullptr, NULL, RRF_RT_ANY, NULL, (LPBYTE)&sValue, &valueSize); //GetLastErrorInfo();

wcout << lResult << endl;

wcout << sValue << endl;

RegCloseKey(regKey);

*/

/*RegOpenKeyEx*/

/*lResult = RegOpenKeyEx(HKEY_LOCAL_MACHINE, TEXT("SYSTEM"), 0, KEY_READ, &hOpenKey);

if (lResult == ERROR_SUCCESS)

wcout << hOpenKey << endl;

else

wcout << "OpenError" << endl;*/

/*RegCreateKey*/

/*

TCHAR sKey[64] = { 0 };

LONG lResult = -1;

HKEY hCreateKey = NULL;

lstrcpy(sKey, TEXT("TEST"));

//lResult = RegCreateKey(HKEY_LOCAL_MACHINE, TEXT("SOFTWARE\\Microsoft\\Windows\\MyKey"), &hCreateKey);

lResult = RegCreateKey(HKEY_LOCAL_MACHINE, TEXT("MyKey"), &hCreateKey);

wcout << GetLastError() << endl;

if (lResult == ERROR_SUCCESS)

{

wcout << TEXT("Create success") << endl;

RegCloseKey(hCreateKey);

}

else

wcout << GetLastError() << endl;

wcout << lResult << endl;

GetLastErrorInfo();

system("PAUSE");

return 0;

*/

实验三系统进程管理

实验目的:

了解进程基本常识

了解进程与线程基本知识

遍历系统中进程,并可以终止指定进程

实验内容:

1.编写代码,获取当前进程信息

2.判断指定程序是否在程序中运行

3.终止指定进程运行

实验步骤:

参考代码。

获得当前进程的列表

//进程描述信息

typedef struct _tagPROCESSINFO

{

DWORD dwPID;

TCHAR strPath[_MAX_PATH];

TCHAR strName[_MAX_FNAME];

} PROCESSINFO, *LPPROCESSINFO;

//获取进程信息列表

BOOL EnumProcessesInfo( PROCESSINFO* lpPsInfo, ULONG ulSize, ULONG* pulNeeded ) // lpPsInfo [out] : 指向PROCESSINFO结构数组的指针

// nSize [in] : lpPsInfo中的元素个数

// nNeeded [out] : 实际的元素个数

// 返回值 : TRUE : 成功; FALSE : 失败

{

ASSERT( pulNeeded );

LPDWORD lpdwPIDs ; //存储进程ID数组

DWORD dwbSize, dwbSize2;

dwbSize2 = 256 * sizeof( DWORD );

lpdwPIDs = NULL;

do {

if( lpdwPIDs ) {

HeapFree( GetProcessHeap(), 0, lpdwPIDs );

dwbSize2 *= 2;

}

lpdwPIDs = (LPDWORD)HeapAlloc( GetProcessHeap(), 0, dwbSize2 ); if( lpdwPIDs == NULL ) {

return FALSE ;

}

if( ! ::EnumProcesses( lpdwPIDs, dwbSize2, &dwbSize ) ) {

HeapFree( GetProcessHeap(), 0, lpdwPIDs ) ;

return FALSE ;

}

}while( dwbSize == dwbSize2 ) ;

ULONG ulCount = dwbSize / sizeof( DWORD );

//如果为询问数量,则返回实际数量

if ( NULL == lpPsInfo && 0 == ulSize ) {

*pulNeeded = ulCount;

return TRUE;

}

ASSERT( lpPsInfo );

if ( NULL == lpPsInfo ) {

return FALSE;

}

if ( ulSize <= ulCount ) {

*pulNeeded = ulSize;

}

else {

*pulNeeded = ulCount;

}

//获得进程信息

HANDLE hProcess;

HMODULE hModule;

DWORD dwSize;

char path_buffer[_MAX_PATH];

char drive[_MAX_DRIVE];

char dir[_MAX_DIR];

char fname[_MAX_FNAME];

char ext[_MAX_EXT];

// Loop through each ProcID.

for( ULONG ulIndex = 0 ; ulIndex < (*pulNeeded) ; ulIndex++ )

{

// Open the process (if we can... security does not

// permit every process in the system).

// TRACE("PID To Open:%d\r\n", lpdwPIDs[ulIndex] );

lpPsInfo[ulIndex].dwPID = lpdwPIDs[ulIndex];

lpPsInfo[ulIndex].strPath[0] = 0;

lpPsInfo[ulIndex].strName[0] = 0;

// Because Can't Open 0 And 8 Process,

// Mark Them At There

if ( 0 == lpdwPIDs[ulIndex] ) {

strcpy( lpPsInfo[ulIndex].strName, "System Idle Process" ); continue;

}

else if ( 8 == lpdwPIDs[ulIndex] ) {

strcpy( lpPsInfo[ulIndex].strName, "System" );

continue;

}

// Open Process And Get Process Infomation

hProcess = OpenProcess(

PROCESS_QUERY_INFORMATION | PROCESS_VM_READ,

FALSE, lpPsInfo[ulIndex].dwPID );

if( hProcess != NULL )

{

// Here we call EnumProcessModules to get only the

// first module in the process this is important,

// because this will be the .EXE module for which we

// will retrieve the full path name in a second.

if( EnumProcessModules( hProcess, &hModule,

sizeof(hModule), &dwSize ) ) {

// Get Full pathname:

if( GetModuleFileNameEx( hProcess, hModule,

path_buffer, sizeof(path_buffer) ) ) {

_tsplitpath( path_buffer, drive, dir, fname, ext ); strcpy( lpPsInfo[ulIndex].strPath, path_buffer );

sprintf( lpPsInfo[ulIndex].strName, "%s%s", fname, ext );

// TRACE( "ModuleFileName:%s\r\n", path_buffer );

}

}

CloseHandle( hProcess ) ;

}

}

return TRUE;

}

实验四系统进程注入

实验目的:

了解动态链接库知识

了解远程注入

利用工具查看注入是否成功

实验内容:

1.编写简单DLL程序

2.编写远程注入程序,主要包括:

●系统提权

●进程遍历

●远程注入

3.判断注入是否成功

实验步骤:

参考代码

door.cpp

#include

#include

#include

BOOL APIENTRY DllMain(HANDLE hModule, DWORD dwReasion, LPVOID lpReserved) {

char* szPID = (char*)malloc(10 * sizeof(char));

switch (dwReasion)

{

Sleep(1000);

case DLL_PROCESS_ATTACH:

{

//itoa(GetCurrentProcessId(), szPID, 10);

sprintf(szPID, "%s", GetCurrentProcessId());

MessageBox(NULL, szPID, "DLL", MB_OK);

break;

}

case DLL_THREAD_ATTACH:

{

itoa(GetCurrentProcessId(), szPID, 10);

MessageBox(NULL, szPID, "DLL", MB_OK);

break;

}

case DLL_PROCESS_DETACH:

{

itoa(GetCurrentProcessId(), szPID, 10);

MessageBox(NULL, szPID, "DLL", MB_OK);

break;

}

case DLL_THREAD_DETACH:

{

itoa(GetCurrentProcessId(), szPID, 10);

MessageBox(NULL, szPID, "DLL", MB_OK);

break;

}

default:

return true;

}

}

Inject.cpp

#include

#include

#include

#include

int EnableDebugPriv(const char* name)

{

HANDLE hToken;

TOKEN_PRIVILEGES tp;

LUID luid;

OpenProcessToken(GetCurrentProcess(),

TOKEN_QUERY | TOKEN_ADJUST_PRIVILEGES, &hToken);

LookupPrivilegeValue(NULL, name, &luid);

tp.PrivilegeCount = 1;

tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;

tp.Privileges[0].Luid = luid;

AdjustTokenPrivileges(hToken, 0, &tp, sizeof(TOKEN_PRIVILEGES), NULL, NULL);

return 0;

}

BOOL InjectDll(const char* DllPath, const DWORD dwRemoteProcessID)

{

HANDLE hRemoteProcess;

char* pszLibFileRemote;

HANDLE hRemoteThread;

DWORD threadID;

EnableDebugPriv(SE_DEBUG_NAME);

printf("Injected ID:%d\n", dwRemoteProcessID);

printf("Inject DLL:%s\n", DllPath);

hRemoteProcess = OpenProcess(PROCESS_ALL_ACCESS, FALSE, dwRemoteProcessID);

pszLibFileRemote = (char*)VirtualAllocEx(hRemoteProcess, NULL,

lstrlen(DllPath) + 1, MEM_COMMIT, PAGE_READWRITE);

WriteProcessMemory(hRemoteProcess, pszLibFileRemote,

(void*)DllPath, lstrlen(DllPath) + 1, NULL);

PTHREAD_START_ROUTINE pfnStartAddr = (PTHREAD_START_ROUTINE)GetProcAddress( GetModuleHandle(TEXT("Kernel32.dll")), "LoadLibraryA");

hRemoteThread = CreateRemoteThread(hRemoteProcess, NULL, 0, pfnStartAddr, pszLibFileRemote, 0, &threadID);

if (hRemoteThread == NULL)

{

printf("DLL Injected Error.\n");

return 0;

}

else

{

printf("DLL Injected success %d.\n", hRemoteThread);

}

printf("%d\n",threadID);

Sleep(100000);

CloseHandle(hRemoteProcess);

CloseHandle(hRemoteProcess);

return true;

}

DWORD GetProcessID()

{

DWORD dwPID = -1;

HANDLE hSnap;

PROCESSENTRY32 pe;

char* targetFile = "notepad.exe";

hSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);

ZeroMemory(&pe, sizeof(pe));

pe.dwSize = sizeof(pe);

Process32First(hSnap, &pe);

if(strstr(targetFile, pe.szExeFile))

{

dwPID = pe.th32ProcessID;

return dwPID;

}

while (true)

{

ZeroMemory(&pe, sizeof(pe));

pe.dwSize = sizeof(pe);

if (!Process32Next(hSnap, &pe))

{

dwPID = -1;

break;

}

if (strstr(targetFile, pe.szExeFile))

{

dwPID = pe.th32ProcessID;

break;

}

}

return dwPID;

}

int main(int argc, char** argv)

{

char myFILE[MAX_PATH];

GetCurrentDirectory(MAX_PATH, myFILE);

strcat(myFILE, "\\door.dll");

InjectDll(myFILE, GetProcessID());

return 0;

}

操作系统实验报告

操作系统实验报告 集团企业公司编码:(LL3698-KKI1269-TM2483-LUI12689-ITT289-

实验二进程调度1.目的和要求 通过这次实验,理解进程调度的过程,进一步掌握进程状态的转变、进程调度的策略,进一步体会多道程序并发执行的特点,并分析具体的调度算法的特点,掌握对系统性能的评价方法。 2.实验内容 阅读教材《计算机操作系统》第二章和第三章,掌握进程管理及调度相关概念和原理。 编写程序模拟实现进程的轮转法调度过程,模拟程序只对PCB进行相应的调度模拟操作,不需要实际程序。假设初始状态为:有n个进程处于就绪状态,有m个进程处于阻塞状态。采用轮转法进程调度算法进行调度(调度过程中,假设处于执行状态的进程不会阻塞),且每过t个时间片系统释放资源,唤醒处于阻塞队列队首的进程。 程序要求如下: 1)输出系统中进程的调度次序; 2)计算CPU利用率。 3.实验环境 Windows操作系统、VC++6.0 C语言 4设计思想: (1)程序中进程可用PCB表示,其类型描述如下:

structPCB_type { intpid;//进程名 intstate;//进程状态 2——表示“执行”状态 1——表示“就绪”状态 0——表示“阻塞”状态 intcpu_time;//运行需要的CPU时间(需运行的时间片个数) } 用PCB来模拟进程; (2)设置两个队列,将处于“就绪”状态的进程PCB挂在队列ready中;将处于“阻塞”状态的进程PCB挂在队列blocked中。队列类型描述如下: structQueueNode{ structPCB_typePCB; StructQueueNode*next; } 并设全程量: structQueueNode*ready_head=NULL,//ready队列队首指针 *ready_tail=NULL,//ready队列队尾指 针

操作系统实验报告4

《操作系统》实验报告 实验序号: 4 实验项目名称:进程控制

Printf(“child Complete”); CloseHandle(pi.hProcess); CloseHandle(pi hThread); ﹜ 修改后: #include #include int main(VOID) { STARTUPINFO si; PROCESS_INFORMA TION pi; ZeroMemory(&si,sizeof(si)); si.cb=sizeof(si); ZeroMemory(&pi,sizeof(pi)); if(!CreateProcess(NULL, "c:\\WINDOWS\\system32\\mspaint.exe", NULL, NULL, FALSE, 0, NULL, NULL, &si,&pi)) { fprintf(stderr,"Creat Process Failed"); return -1; } WaitForSingleObject(pi.hProcess,INFINITE); printf("child Complete"); CloseHandle(pi.hProcess); CloseHandle(pi.hThread); } 在“命令提示符”窗口运行CL命令产生可执行程序4-1.exe:C:\ >CL 4-1.cpp

实验任务:写出程序的运行结果。 4.正在运行的进程 (2)、编程二下面给出了一个使用进程和操作系统版本信息应用程序(文件名为4-5.cpp)。它利用进程信息查询的API函数GetProcessVersion()与GetVersionEx()的共同作用。确定运行进程的操作系统版本号。阅读该程序并完成实验任务。 #include #include

操作系统实验三

计算机操作系统实验报告 实验内容: P、V原语的模拟实现 实验类型:验证型 指导教师:毕国堂 专业班级: 姓名: 学号: 实验地点:东6E507 实验时间:2017/10/23

一、实验目的 1.理解信号量相关理论 2.掌握记录型信号量结构 3.掌握P、V原语实现机制 二、实验内容 1.输入给定的代码 2.进行功能测试并得出证正确结果 三、实验要求 1.分析signal和wait函数功能模块 ●Signal函数 在进行资源增加时,首先判断增加的资源是否存在,如果不存在则报错 并结束函数;如果存在则将需要增加的资源数量加一,然后再判断增加 后的资源数是否大于0,如果大于0则表示之前等待队列为空,没有需 要分配的进程;如果增加后的资源不大于0,表示之前等待队列中存在 进程,则将队首的进程取出并将资源分给该进程。 ●Wait 函数 在执行wait函数时,先判断请求的资源和进程是否存在,如果不存在则 报错提示;如果存在则将对应资源的资源数减一,然后判断减少后的资 源数是否小于0,如果小于0,表示该资源等待队列为空,可直接将资源 分配给请求的进程;如果不小于0则表示之前资源的等待队列不为空, 则将请求的进程插在等待队列最后。 2.画出signal和wait函数流程图

3.撰写实验报告 四、实验设备 1.PC机1台安装visual c++ 6.0 五、测试

1.首先将所有的资源分配完 2.这时再请求资源时就会出现等待现象 3.此时增加一个资源s0,则进程1对s0的等待结束直接获取资源s0 4.当再增加资源s0、s1时则进程1也结束对资源s1的等待,并且s0资源 为有空闲状态 六、实验思考 1.如何修改wait操作,使之能一次申请多个信号量? wait函数传入一个进程号和多个资源名,在wait函数中使用循环依

操作系统实验报告

《操作系统原理》实验报告 实验项目名称:模拟使用银行家算法判断系统的状态 一、实验目的 银行家算法是操作系统中避免死锁的算法,本实验通过对银行家算法的模拟,加强对操作系统中死锁的认识,以及如何寻找到一个安全序列解除死锁。 二、实验环境 1、硬件:笔记本。 2、软件:Windows 7 , Eclipse。 三、实验内容 1.把输入资源初始化,形成资源分配表; 2.设计银行家算法,输入一个进程的资源请求,按银行家算法步骤进行检查; 3.设计安全性算法,检查某时刻系统是否安全; 4.设计显示函数,显示资源分配表,安全分配序列。 四、数据处理与实验结果 1.资源分配表由进程数组,Max,Allocation,Need,Available 5个数组组成; 实验采用数据为下表: 2.系统总体结构,即菜单选项,如下图

实验的流程图。如下图 3.实验过程及结果如下图所示

1.首先输入进程数和资源类型及各进程的最大需求量 2.输入各进程的占有量及目前系统的可用资源数量 3.初始化后,系统资源的需求和分配表 4.判断线程是否安全

5.对线程进行死锁判断 五、实验过程分析 在实验过程中,遇到了不少问题,比如算法无法回滚操作,程序一旦执行,必须直接运行到单个任务结束为止,即使产生了错误,也必须等到该项任务结束才可以去选择别的操作。但总之,实验还是完满的完成了。 六、实验总结 通过实验使我对以前所学过的基础知识加以巩固,也对操作系统中抽象理论知识加以理解,例如使用Java语言来实现银行家算法,在这个过程中更进一步了解了银行家算法,通过清晰字符界面能进行操作。不过不足之处就是界面略显简洁,对于一个没有操作过计算机的人来说,用起来可能还是有些难懂。所以,以后会对界面以及功能进行完善,做到人人都可以看懂的算法。

操作系统实验报告心得体会

操作系统实验报告心得体会 每一次课程设计度让我学到了在平时课堂不可能学到的东西。所以我对每一次课程设计的机会都非常珍惜。不一定我的课程设计能够完成得有多么完美,但是我总是很投入的去研究去学习。所以在这两周的课设中,熬了2个通宵,生物钟也严重错乱了。但是每完成一个任务我都兴奋不已。一开始任务是任务,到后面任务就成了自己的作品了。总体而言我的课设算是达到了老师的基本要求。总结一下有以下体会。 1、网络真的很强大,用在学习上将是一个非常高效的助手。几乎所有的资料都能够在网上找到。从linux虚拟机的安装,到linux的各种基本命令操作,再到gtk的图形函数,最后到文件系统的详细解析。这些都能在网上找到。也因为这样,整个课程设计下来,我浏览的相关网页已经超过了100个(不完全统计)。当然网上的东西很乱很杂,自己要能够学会筛选。 不能决定对或错的,有个很简单的方法就是去尝试。就拿第二个实验来说,编译内核有很多项小操作,这些小操作错了一项就可能会导致编译的失败,而这又是非常要花时间的,我用的虚拟机,编译一次接近3小时。所以要非常的谨慎,尽量少出差错,节省时间。多找个几个参照资料,相互比较,

慢慢研究,最后才能事半功倍。 2、同学间的讨论,这是很重要的。老师毕竟比较忙。对于课程设计最大的讨论伴侣应该是同学了。能和学长学姐讨论当然再好不过了,没有这个机会的话,和自己班上同学讨论也是能够受益匪浅的。大家都在研究同样的问题,讨论起来,更能够把思路理清楚,相互帮助,可以大大提高效率。 3、敢于攻坚,越是难的问题,越是要有挑战的心理。这样就能够达到废寝忘食的境界。当然这也是不提倡熬夜的,毕竟有了精力才能够打持久战。但是做课设一定要有状态,能够在吃饭,睡觉,上厕所都想着要解决的问题,这样你不成功都难。 4、最好在做课设的过程中能够有记录的习惯,这样在写实验报告时能够比较完整的回忆起中间遇到的各种问题。比如当时我遇到我以前从未遇到的段错误的问题,让我都不知道从何下手。在经过大量的资料查阅之后,我对段错误有了一定的了解,并且能够用相应的办法来解决。 在编程中以下几类做法容易导致段错误,基本是是错误地使用指针引起的 1)访问系统数据区,尤其是往系统保护的内存地址写数据,最常见就是给一个指针以0地址 2)内存越界(数组越界,变量类型不一致等) 访问到不属于你的内存区域

操作系统实验指导_源码参考资料

华东交通大学 软件学院 操作系统实验报告 专业: 计算机科学与技术 姓名: 林庆达 学号: 3103005138 2005-6

试验一进程调度 一、实验目的: 编写和调试一个进程调度程序,以加深对进程的概念及进程调度算法的理解。 二、实验内容:以两种典型算法为例说明实现的算法 (一)、最高优先数优先的调度算法 1、实验原理 进程调度算法:采用最高优先数优先 的调度算法(即把处理机分配给优先数最 高的进程)和先来先服务算法。 每个进程有一个进程控制块(PCB) 表示。进程控制块可以包含如下信息:进 程名、优先数、到达时间、需要运行时间、 已用CPU时间、进程状态等等。 进程的优先数及需要的运行时间可以 事先人为地指定(也可以由随机数产生)。 进程的到达时间为进程输入的时间。 进程的运行时间以时间片为单位进 行计算。 每个进程的状态可以是就绪W (Wait)、运行R(Run)、或完成F(Finish) 三种状态之一。 就绪进程获得CPU后都只能运行一 个时间片。用已占用CPU时间加1来表示。 如果运行一个时间片后,进程的已占 用CPU时间已达到所需要的运行时间, 则撤消该进程,如果运行一个时间片后进 程的已占用CPU时间还未达所需要的运 行时间,也就是进程还需要继续运行,此 时应将进程的优先数减1(即降低一级), 然后把它插入就绪队列等待CPU。 每进行一次调度程序都打印一次运 行进程、就绪队列、以及各个进程的PCB, 以便进行检查。 重复以上过程,直到所有进程都完成为止。 2、源代码: #include "stdio.h" #include #include #define getpch(type) (type*)malloc(sizeof(type)) #define NULL 0

操作系统实验报告

操作系统教程实验报告 专业班级 学号 姓名 指导教师

实验一WINDOWS进程初识 1、实验目的 (1)学会使用VC编写基本的Win32 Consol Application(控制台应用程序)。 (2)掌握WINDOWS API的使用方法。 (3)编写测试程序,理解用户态运行和核心态运行。 2、实验内容和步骤 (1)编写基本的Win32 Consol Application 步骤1:登录进入Windows,启动VC++ 6.0。 步骤2:在“FILE”菜单中单击“NEW”子菜单,在“projects”选项卡中选择“Win32 Consol Application”,然后在“Project name”处输入工程名,在“Location”处输入工程目录。创建一个新的控制台应用程序工程。 步骤3:在“FILE”菜单中单击“NEW”子菜单,在“Files”选项卡中选择“C++ Source File”, 然后在“File”处输入C/C++源程序的文件名。 步骤4:将清单1-1所示的程序清单复制到新创建的C/C++源程序中。编译成可执行文件。 步骤5:在“开始”菜单中单击“程序”-“附件”-“命令提示符”命令,进入Windows “命令提示符”窗口,然后进入工程目录中的debug子目录,执行编译好的可执行程序:E:\课程\os课\os实验\程序\os11\debug>hello.exe 运行结果 (如果运行不成功,则可能的原因是什么?) : (2)计算进程在核心态运行和用户态运行的时间 步骤1:按照(1)中的步骤创建一个新的“Win32 Consol Application”工程,然后将清单1-2中的程序拷贝过来,编译成可执行文件。 步骤2:在创建一个新的“Win32 Consol Application”工程,程序的参考程序如清单1-3所示,编译成可执行文件并执行。 步骤3:在“命令提示符”窗口中运行步骤1中生成的可执行文件,测试步骤2中可执行文件在核心态运行和用户态运行的时间。 E:\课程\os课\os实验\程序\os12\debug>time TEST.exe 步骤4:运行结果 (如果运行不成功,则可能的原因是什么?) : 步骤5:分别屏蔽While循环中的两个for循环,或调整两个for循环的次数,写出运行结果。 屏蔽i循环:

操作系统-实验三

操作系统-实验三 文件系统的用户界面 一、实验目的 进一步理解、使用和掌握文件的系统调用、文件的标准子例程,能利用和选择这些基本的文件操作完成复杂的文件处理工作。 二、实验题目 1.编写一个文件复制的C语言程序:分别使用文件的系统调用read(fd, buf, nbytes), write(fd, buf, nbytes)和文件的库函数fread(buf, size, nitems, fp), fwrite(buf, size, nitems, fp),编写一个文件的复制程序(文件大小>1M ),文件可以编一个C 程序来生成,或使用/usr/bin下的二进制执行文件。 调用格式例如: copy file1 file2 #include main(int argc, char*argv[]) { … fd1=open(argv[1], O_RDONLY); //系统调用 creat (argv[2], 0660); fd2=open(argv[2], O_WRONL Y); while((n=read(fd1, buf, BUFSIZE))>0) write(fd2, buf, n); … main带参的调用方法例(含测试时间): time ./mycp infile outfile 流文件的实验程序请参考该程序完成。

上述函数中nbytes, size和nitems都取值为1时(即一次读写一个字节),比较系统调用和流文件两种程序的执行效率。当nbytes取4096字节,size取1字节且nitems取4096时(即一次读写4096字节),再次比较这两种程序的执行效率(文件大小>1M)。如: 创建大文件的方法之一,比如用creat 创建一个新文件,用open写打开该文件,用lseek将写指针移到很远处,写入随便一个字符。比如移动0x100000,用write写个“1”,就会得到一个1M大小的文件。也可到Linux的/usr/bin找一个1~3M左右的大的执行文件。 对于单独使用的速度较快的计算机,文件要10M~100M。 2.编写一个父子进程之间用无名管道进行数据传送的C程序。父进程逐一读出一个文件的内容,并通过管道发送给子进程。子进程从管道中读出信息,再将其写入一个新的文件。程序结束后,对原文件和新文件的内容进行比较。 3.在两个用户的独立程序之间,使用有名管道,重新编写一个C程序,实现题2的功能。 三、源代码 1.编写一个文件复制的C语言程序:分别使用文件的系统调用read(fd, buf, nbytes), write(fd, buf, nbytes)和文件的库函数fread(buf, size, nitems, fp), fwrite(buf, size, nitems, fp),编写一个文件的复制程序。 程序一 #define BUFSIZE 4096 #include #include #include #include int main(int argc, char *argv[]) { printf("这个是一次4096个字节的运行结果:\n");

操作系统实验报告

操作系统实验报告 实验名称: 系统的引导 所在班级: 指导老师: 老师 实验日期: 2014年3 月29 日

一、实验目的 ◆熟悉hit-oslab实验环境; ◆建立对操作系统引导过程的深入认识; ◆掌握操作系统的基本开发过程; ◆能对操作系统代码进行简单的控制,揭开操作系统的神秘面纱。 二、实验容 1. 阅读《Linux核完全注释》的第6章引导启动程序,对计算机和Linux 0.11的引导过程进行初步的了解。 2. 按照下面的要求改写0.11的引导程序bootsect.s。 3. 有兴趣同学可以做做进入保护模式前的设置程序setup.s。 4. 修改build.c,以便可以使用make BootImage命令 5. 改写bootsect.s主要完成如下功能: bootsect.s能在屏幕上打印一段提示信息XXX is booting...,其中XXX是你给自己的操作系统起的名字,例如LZJos、Sunix等。 6. 改写setup.s主要完成如下功能: bootsect.s能完成setup.s的载入,并跳转到setup.s开始地址执行。而setup.s 向屏幕输出一行"Now we are in SETUP"。setup.s能获取至少一个基本的硬件参数(如存参数、显卡参数、硬盘参数等),将其存放在存的特定地址,并输出到屏幕上。setup.s不再加载Linux核,保持上述信息显示在屏幕上即可。 三、实验环境

本实验使用的系统是windows系统或者是Linux系统,需要的材料是osexp。 四、实验步骤 1. 修改bootsect.s中的提示信息及相关代码; 到osexp\Linux-0.11\boot目录下会看到图1所示的三个文件夹,使用UtraEdit 打开该文件。将文档中的98行的mov cx,#24修改为mov cx,#80。同时修改文档中的第246行为图2所示的情形。 图1图2 图3 2. 在目录linux-0.11\boot下,分别用命令as86 -0 -a -o bootsect.obootsect.s和 ld86 -0 -s -obootsectbootsect.o编译和bootsect.s,生成bootsect文件; 在\osexp目录下点击MinGW32.bat依此输入下面的命令: cd linux-0.11 cd boot as86 -0 -a -o bootsect.obootsect.s ld86 -0 -s -o bootsectbootsect.o

操作系统安全实验报告

CENTRAL SOUTH UNIVERSITY 操作系统安全实验报告 学生姓名 班级学号 指导教师 设计时间

《操作系统安全》实验一 ……Windows系统安全设置实验 一、实验目的 1、了解Windows操作系统的安全性 2、熟悉Windows操作系统的安全设置 3、熟悉MBSA的使用 二、实验要求 1、根据实验中的安全设置要求,详细观察并记录设置前后系统的变化,给出分析报告。 2、采用MBSA测试系统的安全性,并分析原因。 3、比较Windows系统的安全设置和Linux系统安全设置的异同。 三、实验内容人 1、配置本地安全设置,完成以下内容: (1)账户策略:包括密码策略(最小密码长度、密码最长存留期、密码最短存留期、强制密码历史等)和账户锁定策略(锁定阈值、锁定时间、锁定计数等) A.账户锁定策略: 账户锁定阈值:指用户输入几次错误的密码后将用户账户锁定,默认为0,代表不锁定 账户锁定时间:指当用户账户被锁定后经过多长时间会自动解锁,0表示只有管理员才能受控解锁 复位账户锁定计数器:指用户由于输入密码错误开始计数时,计数器保持的时间,当时间过后,计数器复位为0. B.密码策略

(2)账户和口令的安全设置:检查和删除不必要的账户(User用户、Duplicate User用户、测试用户、共享用户等)、禁用guest账户、禁止枚举帐号、创建两个管理员帐号、创建陷阱用户(用户名为Administrator、权限设置为最低)、不让系统显示上次登录的用户名。 A.创建用户: 控制面板----管理工具----计算机管理-----本地用户和组-----用户---创建用户B.修改用户权限: 控制面板---用户账户---管理其他账户---更改账户类型 C.禁止枚举账号: 禁止原因:枚举账号指某些具有黑客行为的蠕虫病毒可以通过扫描WindowsXP系统的指定端口,然后通过共享会话猜测管理员系统密码,因此需要禁止枚举账号 方法:本地安全设置-“安全设置”-“本地策略”-“安全选项”-双击对匿名连接的额外限制-不允许枚举SAM账号和共享”

操作系统实验报告三

课程实验报告 课程名称姓名实验名称实验目的及要求 实验3进程并发与同步 1、加深对进程概念的理解,区分进程并发执行与串行执行; 2、掌握进程并发执行的原理,理解进程并发执行的特点; 3、了解fork()系统调用的返回值,掌握用fork()创建进程的方法;熟悉wait、exit等系统调用; 4、能利用相应的系统调用实现进程树与进程间的同 步。 实 验操作系统:linux Un bu ntu 11.10 环 境实验工具:Vmware 实验内容 1、编写一C语言程序,实现在程序运行时通过系统调用fork()创建两个子进程,使父、子三进程并发执行,父亲进程执行时屏幕显示“I am father ”,儿子进 程执行时屏幕显示“ I am son ",女儿进程执行时屏幕显示“ I am daughter ”。 要求多次连续反复运行这个程序,观察屏幕显示结果的顺序,直至出现不一样的情况为止。要求有运行结果截图与结果分析 2、连续4个fork()的进程家族树,family1-1.c 程序清单如下: #in clude main () { fork(); fork(); fork(); fork(); printf( A\n ”); } 请根据程序运行结果,画出进程家族树,并分析原 因。

3、 修改程序1,在父、子进程中分别使用 wait 、exit 等系统调用“实现”其同 步推进,父进程必须等待儿子进程与女儿进程结束, 才可以输出消息。 写出相应的同 步控制,并分析运行结果。 4、 创建一个子进程,并给它加载程序,其功能是调用键盘命令“ ls -I ”,已知 该键盘命令的路径与文件名为: /bin/ls 。父进程创建子进程, 并加载./child2 程序。 写出相应的程序代码并分析程序运行结果。 1、编写一 C 语言程序,实现在程序运行时通过系统调用 fork()创建两个子进 程,使父、子三进程并发执行,父亲进程执行时屏幕显示“ I am father ”, 儿子进程执行时屏幕显示“ I am son ”,女儿进程执行时屏幕显示“ I am daughter "。并且反复的测试,观察每一次的执行的顺序有什么不同 2、修改程序1,在父、子进程中分别使用 wait 、exit 等系统调用“实现”其同 步推进,父进程必须等待儿子进程与女儿进程结束,才可以输出消息。 4、创建一个子进程,并给它加载程序,其功能是调用键盘命令“ ls -I ”,已知 该键盘命令的路径与文件名为: /bin/ls 。父进程创建子进程, 并加载./child2 程序。 法 描 述 及 实 验 步 骤 调 试过 程及实 验结果

操作系统实验报告

操作系统实验报告 Document number:NOCG-YUNOO-BUYTT-UU986-1986UT

许昌学院 《操作系统》实验报告书学号: 姓名:闫金科 班级:14物联网工程 成绩: 2016年02月

实验一Linux的安装与配置 一、实验目的 1.熟悉Linux系统的基本概念,比如Linux发行版、宏内核、微内核等。 2.掌握Linux系统的安装和配置过程,初步掌握Linux系统的启动和退出方 法。 3.熟悉Linux系统的文件系统结构,了解Linux常用文件夹的作用。 二、实验内容 1.从网络上下载VMware软件和两个不同Linux发行版镜像文件。 2.安装VMware虚拟机软件。 3.在VMware中利用第一个镜像文件完成第一个Linux的安装,期间完成网络 信息、用户信息、文件系统和硬盘分区等配置。 4.在VMware中利用第二个镜像文件完成第二个Linux的安装,并通过LILO或 者GRUB解决两个操作系统选择启动的问题。 5.启动Linux系统,打开文件浏览器查看Linux系统的文件结构,并列举出 Linux常用目录的作用。 三、实验过程及结果 1、启动VMware,点击新建Linux虚拟机,如图所示: 2、点击下一步,选择经典型,点击下一步在选择客户机页面选择 Linux,版本选择RedHatEnterpriseLinux5,如图所示: 3、点击下一步创建虚拟机名称以及所要安装的位置,如图所示: 4、点击下一步,磁盘容量填一个合适大小,此处选择默认值大小 10GB,如图所示: 5、点击完成,点击编辑虚拟机设置,选择硬件选项中的CD-ROM (IDE...)选项,在右侧连接中选择“使用ISO镜像(I)”选项,点 击“浏览”,找到Linux的镜像文件,如图所示:

操作系统实验报告

操作系统实验报告 学生学院计算机学院 专业班级计算机科学与技术3班学号3213005910 学生姓名林虹 指导教师丁国芳 2015 年12月15 日

目录 1 实验一进程调度 (1) 2 实验二银行家算法 (16) 3 实验三动态分区分配方式的模拟 (20) 4 实验四仿真各种磁盘调度算法 (26)

实验一进程调度 1. 实验目的 编写并调试一个模拟的进程调度程序,分别采用“短进程优先”、“时间片轮转”、“高响应比优先”调度算法对随机产生的五个进程进行调度,并比较算法的平均周转时间。以加深对进程的概念及进程调度算法的理解。 2. 实验要求 1.每个进程由一个进程控制块(PCB)表示,进程控制块可以包含如下信息:进程 名、优先数(响应比)、到达时间、需要运行时间(进程的长度)、已运行时间、进 程状态等等(可以根据需要自己设定)。 2.由程序自动生成进程(包括需要的数据,要注意数据的合理范围),第一个进程到 达时间从0开始,其余进程到达时间随机产生。 3.采用时间片轮转调度算法时,进程的运行时间以时间片为单位进行计算。 4.每个进程的状态可以是就绪W(Wait)、运行R(Run)、或完成F(Finish)三种 状态之一。 5.每进行一次调度,程序都要输出一次运行结果:正在运行的进程、就绪队列中的进 程、完成的进程以及各个进程的PCB,以便进行检查。 6.最后计算各调度算法的平均周转时间,并进行比较、分析。 3. 实验内容 a.算法原理 (1)短进程优先调度算法 “短进程优先”调度算法的基本思想是把CPU分配给就绪队列中需要时间最短的进程。 (2)时间片轮转算法 将系统中所有的就绪进程按照FCFS原则,排成一个队列。每次调度时将CPU 分派给队首进程,让其执行一个时间片。时间片的长度从几个ms到几百ms。在一个时间片结束时,发生时钟中断。调度程序据此暂停当前进程的执行,将其送到就绪队列的末尾,并通过上下文切换执行当前的队首进程。进程可以未使用完一个时间片,就出让CPU。 (3)高响应比优先算法 HRRN调度策略同时考虑每个作业的等待时间长短和估计需要的执行时间长短,从中选出响应比最高的作业投入执行。 每个作业完成后要打印该作业的开始运行时刻、完成时刻、周转时间和带权周转时间,这一组作业完成后要计算并打印这组作业的平均周转时间、带权平均周转时间。

操作系统实验报告

实验报告 实验课程名称:操作系统 实验地点:南主楼七楼机房 2018—2019学年(一)学期 2018年 9月至 2019 年 1 月 专业: 班级: 学号: 姓名: 指导老师:刘一男

实验一 实验项目:分时系统模拟 实验学时:2实验日期: 2018-10-25 成绩: 实验目的利用程序设计语言模拟分时系统中多个进程按时间片轮转调度算法进行进程调度的过程; 假设有五个进程A,B,C,D,E,它们的到达时间及要求服务的时间分别为:进程名 A B C D E 到达时间0 1 2 3 4 服务时间 4 3 4 2 4 时间片大小为1,利用程序模拟A,B,C,D,E五个进程按时间片轮转的调度及执行过程并计算各进程的周转时间及带权周转时间。 执行过程并计算各进程的周转时间及带权周转时间。 轮转调度:BDACE

(1)修改时间片大小为2,利用程序模拟A,B,C,D,E五个进程按时间片轮转的调度及执行过程并计算各进程的周转时间及带权周转时间。 轮转调度:ADBCE (2)修改时间片大小为4,利用程序模拟A,B,C,D,E五个进程按时间片轮转的调度及执行过程并计算各进程的周转时间及带权周转时间.

顺序:ABCDE 1、思考 时间片的大小对调度算法产生什么影响?对计算机的性能产生什么影响?答:通过对时间片轮转调度算法中进程最后一次执行时间片分配的优化,提出了一种改进的时间片轮转调度算法,该算法具有更好的实时性,同时减少了任务调度次数和进程切换次数,降低了系统开销,提升了CPU的运行效率,使操作系统的性能得到了一定的提高。 A B C D E 时间片为1 周转时间12 9 14 8 13 3 3 3.5 4 3.25 带权周转 时间 时间片为2 周转时间8 12 13 7 13 2 4 3.25 3.5 3.25 带权周转 时间 时间片为4 周转时间 4 6 9 10 13 1 2 2.25 5 3.25 带权周转 时间

操作系统实验实验1

广州大学学生实验报告 1、实验目的 1.1、掌握进程的概念,明确进程的含义 1.2、认识并了解并发执行的实质 2.1、掌握进程另外的创建方法 2.2、熟悉进程的睡眠、同步、撤消等进程控制方法 3.1、进一步认识并发执行的实质 3.2、分析进程竞争资源的现象,学习解决进程互斥的方法 4.1、了解守护进程 5.1、了解什么是信号 5.2、INUX系统中进程之间软中断通信的基本原理 6.1、了解什么是管道 6.2、熟悉UNIX/LINUX支持的管道通信方式 7.1、了解什么是消息 7.2、熟悉消息传送的机理 8.1、了解和熟悉共享存储机制 二、实验内容 1.1、编写一段程序,使用系统调用fork( )创建两个子进程。当此程序运行时,在系统 中有一个父进程和两个子进程活动。让每一个进程在屏幕上显示一个字符:父进程显示'a',子进程分别显示字符'b'和字符'c'。试观察记录屏幕上的显示结果,并分析原因。 1.2、修改上述程序,每一个进程循环显示一句话。子进程显示'daughter …'及 'son ……',父进程显示'parent ……',观察结果,分析原因。 2.1、用fork( )创建一个进程,再调用exec( )用新的程序替换该子进程的内容 2.2、利用wait( )来控制进程执行顺序 3.1、修改实验(一)中的程序2,用lockf( )来给每一个进程加锁,以实现进程之间的互斥 3.2、观察并分析出现的现象 4.1、写一个使用守护进程(daemon)的程序,来实现: 创建一个日志文件/var/log/Mydaemon.log ; 每分钟都向其中写入一个时间戳(使用time_t的格式) ; 5.1、用fork( )创建两个子进程,再用系统调用signal( )让父进程捕捉键盘上来的中断信号(即按^c键);捕捉到中断信号后,父进程用系统调用kill( )向两个子进程发出信号,子进程捕捉到信号后分别输出下列信息后终止: Child process1 is killed by parent! Child process2 is killed by parent! 父进程等待两个子进程终止后,输出如下的信息后终止: Parent process is killed! 5.2、用软中断通信实现进程同步的机理

操作系统实验报告

操作系统实验报告 银行家算法 班级:计算机()班 姓名:李君益 学号:(号) 提交日期: 指导老师: 林穗 一、设计题目 加深了解有关资源申请、避免死锁等概念,并体会和了解死锁和避免死锁的具体实施方法。 要求编写和调试一个系统动态分配资源的简单模拟程序,观察死锁产生的条件,并采用银行家算法,有效的防止和避免死锁的发生。 二、设计要求

内容: 编制银行家算法通用程序,并检测思考题中所给状态的安全性。 要求: (1)下列状态是否安全?(三个进程共享个同类资源) 进程已分配资源数最大需求数 (状态) (状态) (2)考虑下列系统状态 分配矩阵最大需求矩阵可用资源矩阵 问系统是否安全?若安全就给出所有的安全序列。若进程请求(),可否立即分配? 三、设计分析 一.关于操作系统的死锁 .死锁的产生 计算机系统中有许多独占资源,他们在任一时刻只能被一个进程使用,如磁带机,绘图仪等独占型外围设备,或进程表,临界区等软件资源。两个进程同时向一台打印机输出将导致一片混乱,两个进程同时进入临界区将导致数据库错误乃至程序崩溃。正因为这些原因,所有操作系统都具有授权一个进程独立访问某一辞源的能力。一个进程需要使用独占型资源必须通过以下的次序: ●申请资源 ●使用资源 ●归还资源 若申请施资源不可用,则申请进程进入等待状态。对于不同的独占资源,进程等待的方式是有差别的,如申请打印机资源、临界区资源时,申请失败将一位这阻塞申请进程;而申请打开文件文件资源时,申请失败将返回一个错误码,由申请进程等待一段时间之后重试。只得指出的是,不同的操作系统对于同一种资源采取的等待方式也是有差异的。 在许多应用中,一个进程需要独占访问多个资源,而操作系统允许多个进程并发执行共享系统资源时,此时可能会出现进程永远被阻塞的现象。这种现象称为“死锁”。 2.死锁的定义 一组进程处于死锁状态是指:如果在一个进程集合中的每个进程都在等待只能由该集合中的其他一个进程才能引发的时间,则称一组进程或系统此时发生了死锁。 .死锁的防止 .死锁产生的条件: ●互斥条件

操作系统安全实验2实验报告

操作系统安全实验2 一、实验目的 1、了解Linux操作系统的安全性 2、熟悉Linux操作系统的安全设置 3、建立Linux操作系统的基本安全框架 二、实验要求 1、根据实验中的安全设置要求,详细观察并记录设置前后系统的变化,给出分析报告。 2、使用RPM对系统的软件进行管理,验证系统内软件的完整性,并分析结果。 3、比较Windows系统的安全设置和Linux系统安全设置的异同。 三、实验内容 1、账户和口令安全 (1)查看和添加账户 在终端下输入命令:useradd ***,建立一个新账户; cat /etc/shadaw, 查看系统中的账户列表; (2)添加和更改密码:passwd命令 (3)查看Linux系统中是否有用于检测密码安全的黑客技术语字典及密码检测模块:locate pam_cracklib.so dict|grep crack

2.账户安全设置 (1)强制用户首次登陆时修改口令,强制每90天更改一次口令,并提前10天提示:chage命令

(2)账户的禁用与恢复:passwd命令,锁定除root之外的不必要的超级用户 (3)建立用户组,设置用户:groupadd命令、groupmod命令、gpasswd命令 (4)设置密码规则:/etc/login.defs文件编辑修改,设置用户的密码最长使用天数、最小密码长度等

(5)为账户和组相关系统文件加上不可更改属性,防止非授权用户获取权限:chattr命令、 (6)删除用户和用户组:userdel命令、groupdel命令

(7)限制su命令提权:/etc/pam.d/su文件,在头部添加命令:auth required /lib/security/pam_wheel.so group=wheel 这样,只有wheel组的用户可以su到root用户 (8)将用户加入到某个组:usermod命令 (9)确认shadow中的空口令帐号:awk命令

操作系统实验报告.

学生学号0121210680225 实验课成绩 武汉理工大学 学生实验报告书 实验课程名称操作系统 开课学院计算机科学与技术学院 指导老师姓名刘军 学生姓名李安福 学生专业班级软件sy1201 2014 — 2015 学年第一学期

《操作系统》实验教学大纲 课程编号: 课程名称:操作系统/Operating System 实验总学时数:12学时 适应专业:计算机科学与技术、软件工程 承担实验室:计算机科学与技术学院实验中心 一、实验教学的目的和任务 通过实验掌握Linux系统下常用键盘命令、系统调用、SHELL编程、后台批处理和C程序开发调试手段等基本用法。 二、实验项目及学时分配 序号实验项目名称实验学时实验类型开出要求 01 Linux键盘命令和vi 2 设计必开 02 Linux下C编程 2 设计必开 03 SHELL编程和后台批处理 2 设计必开 04 Linux系统调用(time) 2 设计必开 05 Linux进程控制(fork) 4 设计必开 三、每项实验的内容和要求: 1、Linux键盘命令和vi 要求:掌握Linux系统键盘命令的使用方法。 内容:见教材p4, p9, p40, p49-53, p89, p100 2、Linux下的C编程 要求:掌握vi编辑器的使用方法;掌握Linux下C程序的源程序编辑方法;编译、连接和运行方法。 内容:设计、编辑、编译、连接以及运行一个C程序,其中包含键盘输入和屏幕输出语句。 3、SHELL编程和后台批处理 要求:掌握Linux系统的SHELL编程方法和后台批处理方法。 内容:(1) 将编译、连接以及运行上述C程序各步骤用SHELL程序批处理完成,前台运行。 (2) 将上面SHELLL程序后台运行。观察原C程序运行时输入输出情况。 (3) 修改调试上面SHELL程序和C程序,使得在后台批处理方式下,原键 盘输入内容可以键盘命令行位置参数方式交互式输入替代原键盘输入内容, 然后输出到屏幕。 4、Linux系统调用使用方法。

操作系统实验报告1

操作系统 实验报告 班号:1303107 学号:1130310726 姓名:蔡鹏

1.请简述head.s 的工作原理。 head.s实在32位保护模式下运行的。我认为这段程序主要包括两个部分:1.初始化设置。2.任务执行与切换。 初始设置主要包括了:1.设置GDT表2.设置系统定时芯片3. 设置IDT表(0x08时钟中断和0x80系统调用中断)4.切换到任务0执行 任务切换和执行包括了:1.任务0和任务1 , 2.时钟中断, 3.系统中断 两个任务的在LDT中代码段和数据段描述符的内容都设置为:基地址0x0000;段限长值为0x03ff,实际段长度为4MB。因此在线性地址空间中这个?内核?的代码和数据段与任务的代码和数据段都从线性地址0开始并且由于没有采用分页机制,所以他们都直接对应物理地址0开始处。 为了每隔10毫秒切换运行的任务,head.s程序中把定时器芯片8253的通道0设置成每隔10毫秒就向中断控制芯片8259A发送一个时钟中断请求信号。PC机的ROM BIOS开机时已经在8259A中把时钟中断请求信号设置成中断向量8,因此我们需要在中断8的处理过程中执行任务切换操作。任务切换的实现是查看current变量中的当前运行的任务号,如果为0,就利用任务1的TSS选择符作为操作数执行远跳转指令,从而切换到任务1中,否则反之。

每个任务在执行时,会首先把一个字符的ASCII码放入寄存器AL中,然后调用系统中断调用int 0x80,而该系统调用处理过程则会调用一个简单的字符写屏子程序,把寄存器AL中的字符显示在屏幕上,同时把字符显示的屏幕的下一个位置记录下来,作为下一次显示字符用。在显示过一个字符后,任务代码会使用循环语句延迟一段时间,然后又跳转到任务代码开始处继续循环执行,直到运行了10毫秒而发生了定时中断,从而代码会切换到另一个任务执行。对于任务A,寄存器AL中始终存放字符‘A’,而任务B运行时AL中始终存放字符‘B’。因此程序运行时我们将看到一连串的‘A’和一连串的‘B’间隔的连续不断的显示在屏幕上。若出现了一个‘C’,是由于PC机偶然产生了一个不是时钟中断和系统调用中断的其他中断。因为我们已经在程序中给所有其他中断安装了一个默认中断处理程序。当出现一个其他中断时,系统就会运行这个中断处理程序,于是就会在屏幕上显示一个‘C’,然后退出中断。 4.请记录head.s 的内存分布状况,写明每个数据段,代码段,栈段 的起始与终止的内存地址。

Windows操作系统实验三实验报告

Windows操作系统C/C++ 程序实验 姓名:___________________ 学号:___________________ 班级:___________________ 院系:___________________ ______________年_____月_____日

实验三Windows 2000/xp线程同步 一、背景知识 二、实验目的 在本实验中,通过对事件和互斥体对象的了解,来加深对Windows 2000/xp线程同步的理解。 1) 回顾系统进程、线程的有关概念,加深对Windows 2000/xp线程的理解。 2) 了解事件和互斥体对象。 3) 通过分析实验程序,了解管理事件对象的API。 4) 了解在进程中如何使用事件对象。 5) 了解在进程中如何使用互斥体对象。 6) 了解父进程创建子进程的程序设计方法。 三、工具/准备工作 在开始本实验之前,请回顾教科书的相关内容。 您需要做以下准备: 1) 一台运行Windows 2000/xp Professional操作系统的计算机。 2) 计算机中需安装V isual C++ 6.0专业版或企业版。 四、实验内容与步骤 1. 事件对象 清单4-1程序展示了如何在进程间使用事件。父进程启动时,利用CreateEvent() API创建一个命名的、可共享的事件和子进程,然后等待子进程向事件发出信号并终止父进程。在创建时,子进程通过OpenEvent() API打开事件对象,调用SetEvent() API使其转化为已接受信号状态。两个进程在发出信号之后几乎立即终止。 步骤1:登录进入Windows 2000/xp Professional。 步骤2:在“开始”菜单中单击“程序”-“Microsoft V isual Studio 6.0”–“Microsoft V isual C++ 6.0”命令,进入V isual C++窗口。 步骤3:在工具栏单击“打开”按钮,在“打开”对话框中找到并打开实验源程序3-1.cpp。 步骤4:单击“Build”菜单中的“Compile 3-1.cpp”命令,并单击“是”按钮确认。系统

相关主题
文本预览
相关文档 最新文档