青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

李錦俊(mybios)的blog

游戲開發 C++ Cocos2d-x OpenGL DirectX 數學 計算機圖形學 SQL Server

  C++博客 :: 首頁 :: 聯系 :: 聚合  :: 管理
  86 Posts :: 0 Stories :: 370 Comments :: 0 Trackbacks

公告

QQ:30743734
EMain:mybios@qq.com

常用鏈接

留言簿(16)

我參與的團隊

最新隨筆

搜索

  •  

積分與排名

  • 積分 - 373919
  • 排名 - 67

最新評論

閱讀排行榜

評論排行榜

作者:任明漢

下載源代碼

前言

  你可能不希望在發布程序時附帶上一個外部的 DLL,因為可能會有些用戶在無意中把 DLL 刪除了而造成 EXE 不能正確運行,也有可能該 DLL 會被別人拿去使用,也有可能,此 DLL 會成為破解者破解你的程序的突破口。無論出于何種原因,如果你想把一個 DLL 合并到一個 EXE 中的話,本文向你介紹這種方法。


Win32 程序調用 DLL 的機制

   Win32 EXE 在調用一個外部 DLL 中的函數時,首先要調用 LoadLibary 函數來載入此 DLL 到程序的進程地址空間。 如果 LoadLibary 載入此 DLL 成功,將返回一個該 DLL 的句柄。 這個句柄實際上就是該 DLL 在內存中的起始地址。 在載入 DLL 成功后,還必須調用 GetProcAddress 函數來獲取要調用的函數的地址。然后再根據該地址來調用這個函數。
根據上述原理,我們可以把一個 DLL 作為資源文件放到 EXE 文件中,在程序運行時,分配一塊內存,然后將此資源復制到該分配的內存中,并根據該內存地址計算得到相關的導出函數地址,然后,當我們需要調用某一函數時,可以用該函數在內存中的地址來調用它。
程序實現。
  首先,把要合并的 DLL 作為資源加入到項目的資源文件中,然后在程序運行時,從資源中載入該資源,以得到該 DLL 在內存中的位置:
LPVOID sRawDll; // 資源文件在內存中的地址 
HRSRC hRes; 
HMODULE hLibrary; 
HGLOBAL hResourceLoaded; 
char lib_name[MAX_PATH]; 
GetModuleFileName(hInstance, lib_name, MAX_PATH ); // 得到運行程序的名字 
hLibrary = LoadLibrary(lib_name);                  // 就象載入一個 DLL 一樣載入運行程序到內存中 

if (NULL != hLibrary) 
{
	// 得到指定的資源文件在內存中的位置 
	hRes = FindResource(hLibrary, MAKEINTRESOURCE(IDR_DATA1), RT_RCDATA); 
	if (NULL != hRes) 
	{
		// 將資源文件載入內存 
		hResourceLoaded = LoadResource(hLibrary, hRes); 
		if (NULL != hResourceLoaded) 
		{
			// 得到資源文件大小 
			SizeofResource(hLibrary, hRes); 

			// 鎖定資源以得到它在內存中的地址 
			sRawDll = (LPVOID)LockResource(hResourceLoaded); 
		}
	}
	else return 1; 
	FreeLibrary(hLibrary);
}
else return 1; 
然后,從資源中載入 DLL 到內存函數 LoadPbDllFromMemory 將載入 DLL 到內存中, 該函數有兩個參數,第一個參數是指向 DLL 資源在內存中的地址的指針,也就是前面代碼中的 LockResource 函數的返回值。第二個參數是一個空的指針,如果函數 LoadPbDllFromMemory 運行成功,該指針將指向重新組合后的內存中的 DLL 的起始地址。該函數還有一個功能就是如果運行成功,它將手動地用 DLL_PROCESS_ATTACH 參數調用 DLL 的入口函數 DllMain 來初始化該 DLL。除此之外,它還會手動地載入合并的 DLL 的入口表中導入的 DLL 并調整它們在內存中的相對地址。以下是該函數代碼:
DWORD LoadPbDllFromMemory(LPVOID lpRawDll, LPVOID lpImageDll) 
{
	SYSTEM_INFO sSysInfo; 
	PIMAGE_DOS_HEADER dosHeader; 
	PIMAGE_NT_HEADERS pNTHeader; 
	PIMAGE_SECTION_HEADER section; 
	PIMAGE_IMPORT_DESCRIPTOR pImportDesc; 
	PIMAGE_IMPORT_BY_NAME pOrdinalName; 
	PIMAGE_BASE_RELOCATION baseReloc; 
	PDWORD lpLink; 
	unsigned char Protection[4096]; 
	HINSTANCE hDll; 
	WORD i; 
	DWORD ImagePages,fOldProtect,j,MaxLen,HdrLen,Addr1,Addr2,Pg,Pg1,Pg2; 
	char * sDllName; 

	if(NULL == lpRawDll) return 1 ; 

	dosHeader = (PIMAGE_DOS_HEADER)lpRawDll; 

	// Is this the MZ header? 
	if ((TRUE == IsBadReadPtr(dosHeader,sizeof (IMAGE_DOS_HEADER))) ||
				 (IMAGE_DOS_SIGNATURE != dosHeader->e_magic)) 
		return 2; 

	// Get the PE header. 
	pNTHeader = MakePtr(PIMAGE_NT_HEADERS,dosHeader,dosHeader->e_lfanew); 

	// Is this a real PE image? 
	if((TRUE == IsBadReadPtr(pNTHeader,sizeof ( IMAGE_NT_HEADERS))) || 
				( IMAGE_NT_SIGNATURE != pNTHeader->Signature)) 
		return 3 ; 

	if(( pNTHeader->FileHeader.SizeOfOptionalHeader != 
			sizeof(pNTHeader->OptionalHeader)) || 
		(pNTHeader->OptionalHeader.Magic != IMAGE_NT_OPTIONAL_HDR32_MAGIC)) 
		return 4; 

	if (pNTHeader->FileHeader.NumberOfSections < 1) return 5; 

	section = IMAGE_FIRST_SECTION( pNTHeader ); 
	int HeaderSize = sizeof(IMAGE_SECTION_HEADER); 

	// 節頭長度 
	HdrLen = (DWORD)section - (DWORD)dosHeader + 
			HeaderSize * pNTHeader->FileHeader.NumberOfSections; 

	// 找出最大的節的長度,此節一般是代碼所在的節(.text 節) 
	MaxLen = HdrLen; 
	int ii=0; 

	for (i = 0;i<(DWORD)pNTHeader->FileHeader.NumberOfSections;i++)// find MaxLen 
	{
		if(MaxLen < section[i].VirtualAddress + section[i].SizeOfRawData) 
		{
			MaxLen = section[i].VirtualAddress + section[i].SizeOfRawData; 
		}
		if(strcmp((const char *)section[i].Name,".rsrc") == 0) ii=i; 
	}

	GetSystemInfo(&sSysInfo);
	ImagePages = MaxLen / sSysInfo.dwPageSize; 
	if (MaxLen % sSysInfo.dwPageSize) ImagePages++; 

	// 分配所需的內存 
	DWORD NeededMemory = ImagePages * sSysInfo.dwPageSize; 
	lpImageDll = VirtualAlloc(NULL, NeededMemory, MEM_COMMIT, PAGE_EXECUTE_READWRITE); 

	if (lpImageDll == NULL) return 6; // 分配內存失敗 

	MoveMemory( lpImageDll, lpRawDll, HdrLen ); // 復制節頭 

	DWORD OrgAddr = 0; 
	DWORD NewAddr = 0; 
	DWORD Size = 0; 

	// 復制 .text 節數據 
	for (i = 0;i<pNTHeader->FileHeader.NumberOfSections;i++) 
	{
		OrgAddr = (DWORD)lpImageDll + (DWORD)section[i].VirtualAddress; 
		NewAddr = (DWORD)lpRawDll + (DWORD)section[i].PointerToRawData; 
		Size = (DWORD)section[i].SizeOfRawData; 
		MoveMemory((void *)OrgAddr, (void *)NewAddr, Size ); 
	}

	// 把指針指向新的 DLL 映像 
	dosHeader = (PIMAGE_DOS_HEADER) lpImageDll; // Switch to new image 
	pNTHeader = (PIMAGE_NT_HEADERS) ((DWORD)dosHeader + dosHeader->e_lfanew); 
	section = (PIMAGE_SECTION_HEADER) ((DWORD)pNTHeader + sizeof(IMAGE_NT_HEADERS)); 
	pImageBase = (PBYTE)dosHeader; 

	if((ii!=0) && (IsNT()==TRUE)) 
	{
		section[ii].VirtualAddress = section[ii].VirtualAddress + (DWORD)lpRawDll; 
		section[ii].PointerToRawData = section[ii].PointerToRawData + (DWORD)lpRawDll; 
	}

	DWORD importsStartRVA; 

	// Look up where the imports section is (normally in the .idata section) 
	// but not necessarily so. Therefore, grab the RVA from the data dir. 
	importsStartRVA = GetImgDirEntryRVA(pNTHeader,IMAGE_DIRECTORY_ENTRY_IMPORT); 
	if ( !importsStartRVA ) 
	{
		VirtualFree(dosHeader,0, MEM_RELEASE); 
		return 7; 
	}

	pImportDesc = (PIMAGE_IMPORT_DESCRIPTOR) pNTHeader->
		OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress; 

	if(pImportDesc!= 0) 
		pImportDesc = (PIMAGE_IMPORT_DESCRIPTOR) ((DWORD)pImportDesc + (DWORD)dosHeader); 
	else 
	{
		VirtualFree(dosHeader,0, MEM_RELEASE); 
		return 8; 
	}

	while (1) // 處理各入口表中的 DLL 
	{
		// 檢查是否遇到了空的 IMAGE_IMPORT_DESCRIPTOR 
		if ((pImportDesc->TimeDateStamp==0 ) && (pImportDesc->Name==0)) break; 

		// 從磁盤載入必須的 Dll, 
		// 注意,載入的 DLL 是合并的 DLL 的入口表中的 DLL, 
		// 不是合并到 EXE 中的 DLL 
		sDllName = (char *) (pImportDesc->Name + (DWORD)pImageBase); 
		hDll = GetModuleHandle(sDllName); 

		if (hDll == 0 ) hDll = LoadLibrary(sDllName); 

		if (hDll == 0 ) 
		{
			MessageBox(NULL, "Can''t find required Dll",
					"Error in LoadPbDllFromMemory()",0); 
			VirtualFree(dosHeader,0, MEM_RELEASE); 
			return 9; 
		}

		DWORD *lpFuncNameRef = (DWORD *) (pImportDesc->OriginalFirstThunk +
								 (DWORD)dosHeader); 
		DWORD *lpFuncAddr = (DWORD *) (pImportDesc->FirstThunk +
								 (DWORD)dosHeader); 

		while( *lpFuncNameRef != 0) 
		{
			pOrdinalName = (PIMAGE_IMPORT_BY_NAME) (*lpFuncNameRef +
						 (DWORD)dosHeader); 
			DWORD pIMAGE_ORDINAL_FLAG = 0x80000000; 

			if (*lpFuncNameRef & pIMAGE_ORDINAL_FLAG) 
				*lpFuncAddr = (DWORD) GetProcAddress(hDll,
					 (const char *)(*lpFuncNameRef & 0xFFFF)); 
			else
				*lpFuncAddr = (DWORD) GetProcAddress(hDll,
						 (const char *)pOrdinalName->Name); 

			if (lpFuncAddr == 0) 
			{
				VirtualFree(dosHeader,0, MEM_RELEASE); 
				return 10;// Can''t GetProcAddress 
			}

			lpFuncAddr++;
			lpFuncNameRef++;
		}
		pImportDesc++;
	}

	DWORD TpOffset; 
	baseReloc = (PIMAGE_BASE_RELOCATION)((DWORD)pNTHeader->
     OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_BASERELOC].VirtualAddress); 

	if (baseReloc !=0) 
	{
		baseReloc = (PIMAGE_BASE_RELOCATION) ((DWORD)baseReloc + (DWORD)dosHeader); 
		while(baseReloc->VirtualAddress != 0) 
		{
			PWORD lpTypeOffset = (PWORD) ((DWORD)baseReloc +
					 sizeof(IMAGE_BASE_RELOCATION)); 
			while (lpTypeOffset < (PWORD)((DWORD)baseReloc +
						 (DWORD)baseReloc->SizeOfBlock)) 
			{
				TpOffset = *lpTypeOffset & 0xF000; 
				if(TpOffset == 0x3000) 
				{
					lpLink = (PDWORD) ((DWORD)dosHeader +
                                   baseReloc->VirtualAddress +
                                                      (*lpTypeOffset & 0xFFF)); 
					*lpLink = (DWORD)dosHeader + 
                                            (*lpLink) - pNTHeader->OptionalHeader.ImageBase; 
				}
				else
				{
					if (TpOffset != 0) 
					{
						VirtualFree(dosHeader,0, MEM_RELEASE); 
						return 10; 
					}
				}
				lpTypeOffset++;
			}
			baseReloc = (PIMAGE_BASE_RELOCATION)((DWORD)baseReloc + 
				(DWORD)baseReloc->SizeOfBlock); 
		}
	}

	// 取得原始的內存狀態 
	memset(Protection,0,4096);
	for (i = 0;i<=pNTHeader->FileHeader.NumberOfSections;i++) 
	{
		if (i == pNTHeader->FileHeader.NumberOfSections) 
		{
			Addr1 = 0; 
			Addr2 = HdrLen; 
			j = 0x60000000; 
		}
		else
		{
			Addr1 = section[i].VirtualAddress; 
			Addr2 = section[i].SizeOfRawData; 
			j = section[i].Characteristics; 
		}
		Addr2 += Addr1 - 1; 

		Pg1 = Addr1 / sSysInfo.dwPageSize; 
		Pg2 = Addr2 / sSysInfo.dwPageSize; 
		for(Pg = Pg1 ;Pg<=Pg2;Pg++) 
		{
			if (j & 0x20000000) Protection[Pg] |= 1; // Execute 
			if (j & 0x40000000) Protection[Pg] |= 2; // Read 
			if (j & 0x80000000) Protection[Pg] |= 4; // Write 
		}
	}

	// 恢復原始的內存狀態 
	Addr1 = (DWORD)dosHeader; 
	for (Pg = 0 ;Pg<= ImagePages;Pg++) 
	{
		switch(Protection[Pg])
		{
		case 2: 
			fOldProtect = PAGE_READONLY; 
			break;
		case 3: 
			fOldProtect = PAGE_EXECUTE_READ; 
			break;
		case 6: 
			fOldProtect = PAGE_READWRITE; 
			break;
		default: 
			// Ignore strange combinations
			fOldProtect = PAGE_EXECUTE_READWRITE;  
			break;
		}

		if (fOldProtect !=PAGE_EXECUTE_READWRITE) 
		{
			if (VirtualProtect((void *)Addr1, 
				sSysInfo.dwPageSize, 
				fOldProtect,
				&fOldProtect) == 0) 
			{
				VirtualFree(dosHeader,0, MEM_RELEASE); 
				return 11; 
			}
		}
		Addr1 += sSysInfo.dwPageSize; 
	}

	EntryPoint = (LPENTRYPOINT) ((DWORD)pNTHeader->OptionalHeader.AddressOfEntryPoint +
				 (DWORD)dosHeader); 
	LPVOID lpReserved = 0; 
	EntryPoint((HINSTANCE)dosHeader, DLL_PROCESS_ATTACH, lpReserved); 
	lpImageDll2=lpImageDll;
	return 0; 
}
  一但 DLL 被正確地載入到內存中,我們就可以通過自定義函數 GetProcAddressDirectly 來獲取某函數在內存中的地址,并根據該地址來調用該函數,該函數也有兩個參數,第一個參數是指向載入到內存中的 DLL 的起始地址的指針,第二個是要調用的函數的函數名。以下是 GetProcAddressDirectly 函數代碼:
DWORD GetProcAddressDirectly(PIMAGE_DOS_HEADER dosHeader, char * FuncName)  
{ 
	PIMAGE_NT_HEADERS pNTHeader;  
	PIMAGE_EXPORT_DIRECTORY pExportDir;  
	PWORD lpNameOrdinals;  
	LPDWORD lpFunctions;  
	DWORD * lpName;  
	char * lpExpFuncName;  
	DWORD i;  
	DWORD j;  
	char * lpFuncName;  

	if(dosHeader->e_magic != IMAGE_DOS_SIGNATURE) return 0;  

	pNTHeader = (PIMAGE_NT_HEADERS)((DWORD)dosHeader + dosHeader->e_lfanew);  

	if (pNTHeader->Signature != IMAGE_NT_SIGNATURE) return 0;  

	if ((pNTHeader->FileHeader.SizeOfOptionalHeader != sizeof(pNTHeader->OptionalHeader)) ||  
		(pNTHeader->OptionalHeader.Magic != IMAGE_NT_OPTIONAL_HDR32_MAGIC))  
		return 0;  

	DWORD exportsStartRVA, exportsEndRVA;  
	pImageBase = (PBYTE)dosHeader;  

	// Make pointers to 32 and 64 bit versions of the header.  
	pNTHeader = MakePtr( PIMAGE_NT_HEADERS, dosHeader,dosHeader->e_lfanew );  

	exportsStartRVA = GetImgDirEntryRVA(pNTHeader,IMAGE_DIRECTORY_ENTRY_EXPORT);  
	exportsEndRVA = exportsStartRVA +  
		GetImgDirEntrySize(pNTHeader, IMAGE_DIRECTORY_ENTRY_EXPORT);  

	// Get the IMAGE_SECTION_HEADER that contains the exports. This is  
	// usually the .edata section, but doesn''t have to be.  
	PIMAGE_SECTION_HEADER header;  
	header = GetEnclosingSectionHeader( exportsStartRVA, pNTHeader );  
	if ( !header ) return 0;  

	INT delta;  
	delta = (INT)(header->VirtualAddress - header->PointerToRawData);  
	pExportDir = (PIMAGE_EXPORT_DIRECTORY)GetPtrFromRVA(exportsStartRVA, 
				pNTHeader, pImageBase);  


	pExportDir =(PIMAGE_EXPORT_DIRECTORY) (pNTHeader->
	OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_EXPORT].VirtualAddress);  

	if (pExportDir == 0)  
	{ 
		MessageBox(NULL,"Error in GetProcAddressDirectly()",0,0);  
		return 0;  
	} 

	pExportDir =(PIMAGE_EXPORT_DIRECTORY) ((DWORD)pExportDir + (DWORD)dosHeader);  
	lpNameOrdinals =(PWORD)((DWORD)pExportDir->AddressOfNameOrdinals + (DWORD)dosHeader);  
	lpName =(LPDWORD) (pExportDir->AddressOfNames + (DWORD)dosHeader);  
	lpFunctions =(LPDWORD) (pExportDir->AddressOfFunctions + (DWORD)dosHeader);  
	lpFuncName = FuncName;  

	if(HIWORD(lpFuncName)!=0 )  
	{ 
		for( i = 0;i<=pExportDir->NumberOfFunctions - 1;i++)  
		{ 
			DWORD entryPointRVA = *lpFunctions;  

			// Skip over gaps in exported function  
			if ( entryPointRVA == 0 ) continue; 
			for( j = 0;j<=pExportDir->NumberOfNames-1;j++)  
			{ 
				if( lpNameOrdinals[j] == i)  
				{ 
					lpExpFuncName = (char *) (lpName[j] + 
							(DWORD)dosHeader);  
					if(strcmp((char *)lpExpFuncName,(char *)FuncName)==0)  
						return (DWORD) (lpFunctions[i] + 
								(DWORD)dosHeader);  
				} 
			} 
		} 
	} 
	else 
	{ 
		for (i = 0 ;i<=pExportDir->NumberOfFunctions - 1;i++)  
		{ 
			if (lpFuncName == (char *)(pExportDir->Base + i))  
			{ 
				if (lpFunctions[i]) return (unsigned long) (lpFunctions[i] + 
							dosHeader);  
			} 
		} 
	} 
	return 0;  
}


在調用完函數后,不要忘記用 UnloadPbDllFromMemory 來從內存中移去 DLL 以釋放分配的內存,該函數還會用 DLL_PROCESS_DETACH 參數調用 DLL 的入口函數 DllMain 來從調用進程的地址空間卸載該 DLL。 以下是 UnloadPbDllFromMemory 函數代碼:
DWORD UnloadPbDllFromMemory(PIMAGE_DOS_HEADER dosHeader) 
{
	PIMAGE_NT_HEADERS pNTHeader; 
	pNTHeader = (PIMAGE_NT_HEADERS)((DWORD)dosHeader + (DWORD)dosHeader->e_lfanew); 
	EntryPoint = (LPENTRYPOINT)(pNTHeader->OptionalHeader.AddressOfEntryPoint +
					 (DWORD)dosHeader); 
	EntryPoint((HINSTANCE)dosHeader, DLL_PROCESS_DETACH, 0); 
	return VirtualFree(dosHeader, 0, MEM_RELEASE); 
}


關于示例代碼的說明

  在本文附帶的示例代碼中,合并了一個名為 hardware.dll 的動態連接庫,該動態連接庫是一個獲取系統硬件信息的庫文件,其中包括了以下函數:

   getmac 		取得網卡 MAC 
   VolumeNumber 	取得硬盤卷標 
   changeres            改變屏幕分辯率 
   IsDiskInDrive        檢查軟驅中是否插有盤 
   DPGetDefaultPrinter  取得默認的打印機名 
   DPSetDefaultPrinter  設置默認的打印機 
   getserial            取得硬盤的出廠序列號 
   getmetric            取得顯示分辯率 
   PrintStringDirect    直接向打印機發送一個串 
   vfpbeep              讓 PC 喇叭發聲 
   getcpuid             取得 CPU ID 
   getbios              取得主板 BIOS ID 
  在示例代碼中,只調用了其中一個函數 getbios 來獲取主板 BIOS ID, 這里說明一下,該函數實際上好象只能檢測 AWARD 主板的 BIOS, 也就是說它是讀取的是系統內存 0x000fex71 處的值。因為其它牌子的主板 BIOS 的位置稍有不同,但在程序中沒有進行這方面的處理,所以在讀其它牌子的主板 BIOS 時可能會有些問題(讀出的內容可能不正確)。關于此 DLL 的內容和實現,也許我會在另一篇文章中論及。
 

局限
  
在我進行測試時,發現對于有些含有資源的 DLL,在 9x 平臺下可能會有問題。

題外話
  
另外,其它一些本文未提及的非主要的函數,請自行參見源代碼中的注釋。
再,本文涉及 PE 文件格式方面的知識,它們已經超出了本文的范圍,具體信息可參見 MSDM 中的:

  • Peering Inside the PE: A Tour of the Win32 Portable Executable File Format 一文和
  • Microsoft Portable Executable and Common Object File Format Specification 一文
  • 特別感謝盧春明(Aming)在我編寫本文時所作的一些技術方面的建議和指導


    如果本文對你的開發有所幫助,并且你手頭恰好有零錢。

    不如打賞我一杯咖啡,鼓勵我繼續分享優秀的文章。




    posted on 2006-11-21 00:45 李錦俊(mybios) 閱讀(9645) 評論(6)  編輯 收藏 引用 所屬分類: C++

    Feedback

    # re: 【轉貼】直接載入內存中的DLL 2006-12-10 10:33 表弟(鋒)
    這是什么來的表哥?很多都看不懂哦!  回復  更多評論
      

    # re: 【轉貼】直接載入內存中的DLL 2006-12-10 19:47 李錦俊
    @表弟(鋒)
    等你長大了就能看懂的了.現在你還小.  回復  更多評論
      

    # re: 【轉貼】直接載入內存中的DLL 2007-02-13 11:51 As
    有必要嗎?
    不如不要使用DLL,直接寫代碼得拉,
    (這樣不是更帥,讓別人直接破解PE結構)

    程序的路不好走~~,繼續加油~~
      回復  更多評論
      

    # re: 【轉貼】直接載入內存中的DLL 2008-06-18 16:00 sdfs
    不太清楚這樣做的意義,把dll drop出來不是更方便?  回復  更多評論
      

    # re: 【轉貼】直接載入內存中的DLL 2010-01-15 08:55 dibaplus@gmail.com
    很好的文章,推薦  回復  更多評論
      

    # re: 【轉貼】直接載入內存中的DLL 2010-02-22 23:54 表弟。鋒
    @表弟(鋒)
    等你長大了就能看懂的了.現在你還小.

    雖然現在還是看不明,但系突然看到這條留言我覺得超意外的,哈哈  回復  更多評論
      

    青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            久久视频国产精品免费视频在线| 国产精品性做久久久久久| 国产一区二区三区丝袜| 翔田千里一区二区| 久久午夜精品一区二区| 欧美影院在线播放| 欧美精品1区2区| 国产欧美精品日韩精品| 久久婷婷影院| 亚洲国产婷婷综合在线精品 | 欧美一区二区三区在线看 | 亚洲美女精品一区| 久久福利精品| 在线看日韩欧美| 午夜在线a亚洲v天堂网2018| 香蕉视频成人在线观看 | 欧美激情一区二区三区不卡| 国产欧美日本一区视频| 久久伊人精品天天| 欧美黄色一区二区| 在线观看视频一区| 亚洲激情午夜| 久久人人九九| 亚洲视频网站在线观看| 午夜天堂精品久久久久| 亚洲激情成人网| 亚洲一区二区三区四区五区午夜| 欧美国产亚洲精品久久久8v| 国内精品久久久久久| 欧美激情小视频| 久久夜色精品一区| 亚洲无毛电影| 久久综合色88| 在线看日韩欧美| 国产精品99久久99久久久二8 | 性欧美xxxx视频在线观看| 欧美在线视频不卡| 亚洲色在线视频| 最新国产精品拍自在线播放| 国产精品日日摸夜夜摸av| 欧美88av| 国产一区二区无遮挡| 一区二区三区视频在线| 亚洲激情av| 最近中文字幕mv在线一区二区三区四区| 久久激情五月婷婷| 国产一区二区三区直播精品电影 | 一区二区电影免费在线观看| 欧美日韩精品| 亚洲视频第一页| 久久综合色影院| 久久久久久久一区二区三区| 久久精品99国产精品酒店日本| 国产欧美日韩在线播放| 久久精品电影| 国产精品蜜臀在线观看| 亚洲精品影视在线观看| 国产精品vip| 亚洲人成欧美中文字幕| 欧美日韩中文在线观看| 午夜精品在线观看| 欧美日韩成人综合在线一区二区| 一本大道久久a久久精品综合| 久久久久成人精品| 亚洲精品日韩一| 久久婷婷人人澡人人喊人人爽| 久久精品亚洲精品| 国产性色一区二区| 性感少妇一区| 亚洲美女啪啪| 亚洲在线不卡| 在线日韩中文字幕| 久久综合久久美利坚合众国| 蜜臀91精品一区二区三区| 精品成人在线观看| 亚洲激情av在线| 亚洲精选视频在线| 午夜精品电影| 亚洲精品资源| 欧美激情一区在线| 这里只有精品视频| 欧美在线亚洲一区| 欧美激情按摩| 久久嫩草精品久久久精品| 欧美国产一区视频在线观看| 亚洲国产精品第一区二区| 99国内精品久久| 久久九九有精品国产23| 亚洲视频在线观看免费| 欧美影院久久久| 欧美1级日本1级| 国产欧美精品日韩区二区麻豆天美| 欧美电影电视剧在线观看| 亚洲精品日韩久久| 久久aⅴ国产紧身牛仔裤| 亚洲一区二区三区精品在线| 国产精品久久久久久亚洲毛片| 亚洲欧美在线x视频| 狼人天天伊人久久| 国产精品影片在线观看| 亚洲精品中文在线| 亚洲国产婷婷| 欧美视频福利| 亚洲精品乱码久久久久久黑人| 亚洲专区一二三| 欧美日韩国产精品自在自线| 欧美成人精品在线视频| 国产在线拍偷自揄拍精品| 免费亚洲电影在线| 亚洲一区在线免费| 亚洲欧美文学| 亚洲激精日韩激精欧美精品| 国产精品免费一区二区三区在线观看| 久久久999精品免费| 在线综合欧美| 亚洲韩国青草视频| 久久久久9999亚洲精品| 国产日韩在线视频| 欧美日韩免费在线观看| 久久影音先锋| 亚洲欧美另类在线| 久久成人亚洲| 国产自产精品| 国产精品美女在线观看| 欧美久久成人| 一本色道久久综合狠狠躁篇的优点| 日韩午夜激情av| 欧美日韩国产三区| 一本色道久久综合亚洲精品高清| 一区二区免费在线播放| 欧美欧美在线| 久久午夜激情| 久久国产欧美精品| 性刺激综合网| 欧美成人自拍| 另类图片国产| 麻豆久久精品| 美女诱惑一区| 免费一级欧美片在线观看| 亚洲精选中文字幕| 欧美视频一区在线观看| 亚洲一区亚洲| 亚洲在线一区二区| 欧美h视频在线| 9国产精品视频| 亚洲精品一区二区三区蜜桃久| 欧美日韩亚洲91| 欧美精选在线| 欧美在线观看日本一区| 欧美亚洲网站| 亚洲国产成人精品视频| 亚洲高清久久| 欧美在线精品一区| 欧美一区二区三区精品电影| 在线观看视频一区| 亚洲高清免费| 国产精品普通话对白| 国产精品五区| 欧美成人精品不卡视频在线观看| 蜜桃av综合| 欧美日韩国产成人精品| 欧美性生交xxxxx久久久| 国产精品亚洲一区二区三区在线| 国产精品乱码妇女bbbb| 国产一区二区三区四区五区美女| 国内偷自视频区视频综合| 欧美三级欧美一级| 国产精品嫩草久久久久| 欧美二区在线看| 欧美一区午夜视频在线观看| 亚洲美女视频网| 午夜精品久久久久久久99樱桃| 欧美在线观看天堂一区二区三区| 亚洲毛片在线免费观看| 另类综合日韩欧美亚洲| 亚洲黄色性网站| 免费在线看一区| 一本色道久久综合亚洲精品小说| 欧美韩日精品| 久久亚洲私人国产精品va媚药| 欧美激情第一页xxx| 麻豆精品一区二区av白丝在线| 午夜精品理论片| 欧美xart系列高清| 在线一区日本视频| 日韩亚洲欧美成人一区| 亚洲国产精品综合| 久久天天狠狠| 日韩视频在线永久播放| 91久久精品国产91久久| 亚洲欧美日韩天堂| 亚洲综合电影| 在线中文字幕一区| 开心色5月久久精品| 国产精品一区二区在线观看不卡| 欧美日韩精品一区| 欧美日韩高清在线| 伊人狠狠色丁香综合尤物| 亚洲一区二区免费在线| 亚洲午夜成aⅴ人片|