久久福利_99r_国产日韩在线视频_直接看av的网站_中文欧美日韩_久久一

您的位置:首頁技術文章
文章詳情頁

Android Binder的原理與使用

瀏覽:3日期:2022-09-19 13:45:55
前言

Binder是安卓中實現IPC(進程間通信的)常用手段,四大組件之間的跨進程通信也是利用Binder實現的,Binder是學習四大組件工作原理的的一個重要基礎。 好多文章都會深入C代碼去介紹Binder的工作流程,沒點水平真的難以理解,本文不會太深入底層去剖析原理,盡可能較為簡單的讓大家了解Binder是怎么工作的。

Binder的使用

在介紹Binder原理之前,我們先來看看在安卓中怎么使用Binder來進程間通信。 在使用之前我們先來介紹Binder的幾個方法:

public final boolean transact(int code, Parcel data, Parcel reply, int flags)

protected boolean onTransact(int code, Parcel data, Parcel reply, int flags)

這兩個方法分別代表了客戶端和服務端,transact用來發送消息,onTransact負責接收transact傳過來的消息,這一點很容易理解。

code 方法標識符,在相同進程中,我們很容易的通過方法調用來執行我們的目標方法,但是在不同的進程間,方法調用的方式就不能再用了,所以我們使用code來表示遠程調用函數的標識。這個標識必須介于FIRST_CALL_TRANSACTION(0x00000001)和LAST_CALL_TRANSACTION(0x00ffffff)之間。 data Parcel類型的數據包,要傳給客戶端的請求參數。 reply 如果客戶端需要返回值,則reply就是服務端返回的數據。 flags 用來區分這個調用是普通調用還是單程調用,普通調用時,Client端線程會阻塞,直到從Server端接收到返回值,若flag==IBinder.FLAG_ONEWAY,則這次調用是單程調用,Client在傳出數據后會立即執行下一段代碼,此時兩端異步執行,單程調用時函數返回值必須為void (也就是單程調用必須舍棄返回值,要返回值就必須阻塞等待)

利用這兩個方法我們就可以實現Client和Server端的通信,接下來我們看看具體該怎么使用。 在Server接收到Client傳來的消息(data)時,會對data進行驗證data.enforceInterface(DESCRIPTOR),DESCRIPTOR是一個字符串類型的描述符,當data的描述符跟DESCRIPTOR相同時才能通過驗證。

public class Stub extends Binder { //描述符 public static final java.lang.String DESCRIPTOR = 'MyTestBinder'; //code 方法描述符 public static final int TRANSACTION_test0 = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0); public static final int TRANSACTION_test1 = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1); @Override protected boolean onTransact(int code, Parcel data, Parcel reply, int flags) throws RemoteException {switch (code) { case TRANSACTION_test0://驗證描述符data.enforceInterface(DESCRIPTOR);//執行方法test0();return true; case TRANSACTION_test1://驗證描述符data.enforceInterface(DESCRIPTOR);//執行方法test1(data, reply);return true;}return super.onTransact(code, data, reply, flags); } //無返回值 private void test0() {Log.d('MyBinderServer', 'test0'); } //有返回值 private void test1(Parcel data, Parcel reply) {Log.d('MyBinderServer', 'test1');reply.writeInt(data.readInt() + 1); }}

我們知道,要想實現Client和Server端的通信連接,就必須讓client知道server端的地址,就跟Http請求,我們要知道服務端的ip和端口。Binder通信其實也是一樣的,那么我們怎么讓Client拿到Server的地址呢? 一種是跟Http請求一樣,我們知道Http請求要把域名轉換成ip和端口,這就是DNS,我們也需要一個Binder的DNS。安卓中也為我們提供了Binder的“DNS”那就是ServiceManager,ServiceManager中注冊了所有系統服務(如MediaServer等),我們可以使用ServiceManager拿到遠程的Binder地址,這種方式叫做有名Binder查找(有名Binder,如MediaServer等這些系統服務被注冊的時候都是有名字的,比如,我們通過WINDOW_SERVICE這個名字就能拿到WindowManager)。但是問題是向ServiceManager注冊服務的過程是系統進程實現的,我們的應用進程不能注冊自己的Binder。 另一種就是利用有名的Binder來輔助傳遞匿名的Binder,也就是說如果有某個有名Binder服務它提供了傳遞Binder的方法,那么我們就可以通過這個Binder服務來傳遞我們的匿名Binder,我們查找到這個有名的Binder是不是就能拿到我們的匿名Binder。正好AMS其實提供了這樣的功能,它通過Service.onBind把匿名的Binder封裝在了Service里面供我們調用。

public class MyService extends Service {@Override public IBinder onBind(Intent intent) {return new Stub(); }}

我們使用binderService()來獲取遠程的Binder。

Intent serviceIntent = new Intent(this, MyService.class);bindService(serviceIntent, new ServiceConnection() { @Override public void onServiceConnected(ComponentName name, IBinder service) {//service可以理解為是遠程Binder的地址,我們利用他跟遠程通信,C++層會轉換這個IBinder跟Binder進行通信 } @Override public void onServiceDisconnected(ComponentName name) { }}, BIND_AUTO_CREATE);

獲取到Binder之后我們補充好通信的代碼:

bindService(serviceIntent, new ServiceConnection() { @Override public void onServiceConnected(ComponentName name, IBinder service) {Parcel data0 = Parcel.obtain();//請求參數Parcel reply0 = Parcel.obtain();//響應參數Parcel data1 = Parcel.obtain();Parcel reply1 = Parcel.obtain();//調用第一個方法try { //添加描述符 data0.writeInterfaceToken(Stub.DESCRIPTOR); /* * 寫入參數,要想傳遞多個int參數,順序調用writeInt * data0.writeInt(10); * data0.writeInt(20); * 獲取 * int num10 = data0.readInt(); * int num20 = data0.readInt(); */ data0.writeInt(10); //發起遠程調用 service.transact(Stub.TRANSACTION_test0, data0, reply0, 0); reply0.readException();} catch (RemoteException e) { e.printStackTrace();} finally { data0.recycle(); reply0.recycle();}//調用第二個方法try { //添加描述符 data1.writeInterfaceToken(Stub.DESCRIPTOR); data1.writeInt(10); //發起遠程調用 service.transact(Stub.TRANSACTION_test1, data1, reply1, 0); reply1.readException(); //讀取返回值 int num = reply1.readInt(); Log.d(TAG, 'reply value: ' + num);} catch (RemoteException e) { e.printStackTrace();} finally { data1.recycle(); reply1.recycle();} } @Override public void onServiceDisconnected(ComponentName name) { }}, BIND_AUTO_CREATE);

為了方便調用,我們寫一個代理類來封裝通信過程

public class Proxy { //描述符 public static final java.lang.String DESCRIPTOR = 'MyTestBinder'; //code 方法描述符 public static final int TRANSACTION_test0 = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0); public static final int TRANSACTION_test1 = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1); private IBinder mRemote; public Proxy(IBinder remote) {this.mRemote = remote; } public void test1() {Parcel data0 = Parcel.obtain();//請求參數Parcel reply0 = Parcel.obtain();//響應參數//調用第一個方法try { //添加描述符 data0.writeInterfaceToken(DESCRIPTOR); /* * 寫入參數,要想傳遞多個int參數,順序調用writeInt * data0.writeInt(10); * data0.writeInt(20); * 獲取 * int num10 = data0.readInt(); * int num20 = data0.readInt(); */ data0.writeInt(10); //發起遠程調用 mRemote.transact(TRANSACTION_test0, data0, reply0, 0); reply0.readException();} catch (RemoteException e) { e.printStackTrace();} finally { data0.recycle(); reply0.recycle();} } public int test2() {Parcel data1 = Parcel.obtain();Parcel reply1 = Parcel.obtain();//調用第二個方法int num = 0;try { //添加描述符 data1.writeInterfaceToken(DESCRIPTOR); data1.writeInt(10); //發起遠程調用 mRemote.transact(TRANSACTION_test1, data1, reply1, 0); reply1.readException(); //讀取返回值 num = reply1.readInt();} catch (RemoteException e) { e.printStackTrace();} finally { data1.recycle(); reply1.recycle();}return num; }}

bindService(serviceIntent, new ServiceConnection() { @Override public void onServiceConnected(ComponentName name, IBinder service) {Proxy proxy = new Proxy(service);proxy.test1();int i = proxy.test2(); } @Override public void onServiceDisconnected(ComponentName name) { }}, BIND_AUTO_CREATE);模糊進程間調用

前邊就是Binder的使用方式,但是至此還遺留了一個問題,我們的Service只有指定了新的進程名之后才會是遠程調用,如果通過bindService 傳遞過來的IBinder對象是同進程的,那我們就不需要使用IBinder.transact進行內核通信了。我們知道同進程之間利用方法調用方式就可以做到通信。 我們在onServiceConnected打印IBinder類型,如果發現是遠程調用,傳給我們的 iBinder 是 BinderProxy 類型,BinderProxy是Binder的內部類一樣實現了IBinder接口,他在native層會對應一個C++的BpBinder,BpBinder 最終會通過Binder驅動跟Server端通信。如果是本地調用,打印出的類型為Stub,說明本地調用時,onServiceConnected傳過來的就是我們在Service的onBinde方法返回的Stub對象本身。基于這個原理,我們可以設計一個轉換方法。

首先我們要怎么判斷當前是遠程調用還是同進程調用呢? 我們使用queryLocalInterface(DESCRIPTOR)方法,Binder中queryLocalInterface不會返回空,而在BinderProxy的實現中,queryLocalInterface返回為null。 Binder:

public IInterface queryLocalInterface(String descriptor) {if (mDescriptor != null && mDescriptor.equals(descriptor)) { return mOwner;}return null; }

mOwner是attachInterface方法傳進來的接口本身,后面還會出現這個方法。 BinderProxy:

public IInterface queryLocalInterface(String descriptor) {return null; }

當發現是遠程調用時我們創建上邊的Proxy來代理跨進程通信過程。要是本地調用我們直接返回本地Stub對象。

public static IMyInterface asInterface(IBinder iBinder){if ((iBinder == null)) { return null;}//獲取本地interfaceIInterface iin = iBinder.queryLocalInterface(DESCRIPTOR);if (((iin != null) && (iin instanceof IMyInterface))) { //不為空,說明是本地調用,直接強轉后返回。 //IMyInterface封裝了test0()、test1()兩個方法,本地對象和Proxy都繼承自接口 return ((IMyInterface)iin );}//為null,遠程調用,新建代理return new Proxy(iBinder); }

把上面相關代碼完善之后

public interface IBinderTest extends android.os.IInterface { /** * 本地Stub對象 */ public static abstract class Stub extends android.os.Binder implements IBinderTest {private static final java.lang.String DESCRIPTOR = 'com.XXX.XXXX.IBinderTest';public Stub() { //綁定DESCRIPTOR,并設置queryLocalInterface方法返回的mOwner this.attachInterface(this, DESCRIPTOR);}/** * 本地遠程轉換 */public static IBinderTest asInterface(android.os.IBinder obj) { if ((obj == null)) {return null; } android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR); if (((iin != null) && (iin instanceof IBinderTest))) {return ((IBinderTest) iin); } return new IBinderTest.Stub.Proxy(obj);}@Overridepublic android.os.IBinder asBinder() { return this;}/** * 處理Client調用請求 */@Overridepublic boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException { java.lang.String descriptor = DESCRIPTOR; switch (code) {case INTERFACE_TRANSACTION: { reply.writeString(descriptor); return true;}case TRANSACTION_testVoidAidl: { data.enforceInterface(descriptor); this.testVoidAidl(); reply.writeNoException(); return true;}case TRANSACTION_testStringAidl: { data.enforceInterface(descriptor); java.lang.String _result = this.testStringAidl(); reply.writeNoException(); reply.writeString(_result); return true;}default: { return super.onTransact(code, data, reply, flags);} }}/** * 遠程調用代理類 */private static class Proxy implements IBinderTest { private android.os.IBinder mRemote; Proxy(android.os.IBinder remote) {mRemote = remote; } @Override public android.os.IBinder asBinder() {return mRemote; } public java.lang.String getInterfaceDescriptor() {return DESCRIPTOR; } @Override public void testVoidAidl() throws android.os.RemoteException {android.os.Parcel _data = android.os.Parcel.obtain();android.os.Parcel _reply = android.os.Parcel.obtain();try { _data.writeInterfaceToken(DESCRIPTOR); mRemote.transact(Stub.TRANSACTION_testVoidAidl, _data, _reply, 0); _reply.readException();} finally { _reply.recycle(); _data.recycle();} } @Override public java.lang.String testStringAidl() throws android.os.RemoteException {android.os.Parcel _data = android.os.Parcel.obtain();android.os.Parcel _reply = android.os.Parcel.obtain();java.lang.String _result;try { _data.writeInterfaceToken(DESCRIPTOR); mRemote.transact(Stub.TRANSACTION_testStringAidl, _data, _reply, 0); _reply.readException(); _result = _reply.readString();} finally { _reply.recycle(); _data.recycle();}return _result; }}/** * 調用函數code */static final int TRANSACTION_testVoidAidl = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);static final int TRANSACTION_testStringAidl = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1); } public void testVoidAidl() throws android.os.RemoteException; public java.lang.String testStringAidl() throws android.os.RemoteException;}

如果你用過AIDL并且看過AIDL生成的代碼,你就會發現上面代碼就是AIDL生成的。 換掉Service的調用

public class MyService extends Service { private String TAG = 'MyService'; @Override public IBinder onBind(Intent intent) {return new MyBinder(); } class MyBinder extends IBinderTest.Stub{@Overridepublic void testVoidAidl() throws RemoteException { Log.d(TAG, 'testVoidAidl');}@Overridepublic String testStringAidl() throws RemoteException { Log.d(TAG, 'testStringAidl'); return 'hello';} }}

以上就是Binder的使用方法以及原理的簡單介紹。

Binder原理

之前介紹了Binder的基本使用,接下來我們來看下Binder的底層原理。

Android Binder的原理與使用

(圖片來源gityuan.com/2015/10/31/…),安卓的應用內存是隔離的,但是內核空間是共享的,我們要實現IPC就要靠共享的內核空間來交換數據。

Android Binder的原理與使用

Binder通信模型:

Android Binder的原理與使用

ioctl

Binder的通信原理:

Android Binder的原理與使用

由于用戶空間的隔離機制(沙盒模式),所以我們要利用內核空間進行IPC操作,用戶空間與內核空間的交互使用的是linux內核的ioctl函數,接下來簡單了解一下ioctl的使用。 ioctl可以控制I/O設備 (驅動設備),提供了一種獲得設備信息和向設備發送控制參數的手段。簡單來說就是使用ioctl可以對驅動設備進行操作。由于我們IPC的過程中內核空間使用虛擬驅動設備/dev/binder控制通信過程,我們要跟這個Binder驅動設備交互就要使用ioctl命令。 簡單介紹下,Linux要實現驅動設備的使用需要三個角色

應用程序(調用方),使用ioctl來發送操作指令。 驅動程序,用來處理ioctl傳來的指令,并完成對驅動設備的操作。驅動程序被注冊進內核之后,就會等待應用程序的調用。 驅動設備,在Binder機制中,驅動設備是/dev/binder,這個文件被映射到每個系統Service的虛擬內存中(后邊會講到),驅動程序可以操作這個文件進行進程間數據交互。

下圖是Linux中應用程序是怎么通過驅動來操作硬件設備的:

Android Binder的原理與使用

來個圖來說一下應用層與驅動程序函數的ioctl之間的聯系:

Android Binder的原理與使用

簡單介紹一下函數:

int (*ioctl) (struct inode * node, struct file *filp, unsigned int cmd, unsigned long arg);

參數:

inode和file:ioctl的操作有可能是要修改文件的屬性,或者訪問硬件。要修改文件屬性的話,就要用到這兩個結構體了,所以這里傳來了它們的指針。 cmd:命令,接下來會講 arg:參數,接下來會講

返回值: 如果傳入的非法命令,ioctl返回錯誤號-EINVAL。 內核中的驅動函數返回值都有一個默認的方法,只要是正數,內核就會傻乎乎的認為這是正確的返回,并把它傳給應用層,如果是負值,內核就會認為它是錯誤了。

ioctl的cmd cmd就是一個數,如果應用層傳來的數值在驅動中有對應的操作,那么就執行,就跟IBinder的transact方法中函數標識是一個道理. 要先定義個命令,就用一個簡單的0,來個命令的頭文件,驅動和應用函數都要包含這個頭文件:

/*test_cmd.h*/#ifndef _TEST_CMD_H#define _TEST_CMD_H#define TEST_CLEAR 0/*定義的cmd*/#endif /*_TEST_CMD_H*/

驅動實現ioctl: 命令TEST_CLEAR的操作就是清空驅動中的kbuf。

int test_ioctl (struct inode *node, struct file *filp, unsigned int cmd, uns igned long arg){ int ret = 0; struct _test_t *dev = filp->private_data; switch(cmd){case TEST_CLEAR:memset(dev->kbuf, 0, DEV_SIZE);dev->cur_size = 0;filp->f_pos = 0;ret = 0;break;default: /*命令錯誤時的處理*/P_DEBUG('error cmd!n');ret = - EINVAL;break; } return ret;}

然后在應用程序中調用ioctl(fd, TEST_CLEAR);就可以執行驅動程序中的清除kbuf的方法。

ioctl的arg ioctl命令還可以傳遞參數,應用層的ioctl(fd,cmd,...)后面的“...”是指可以傳任意類型的一個參數,注意是一個不是任意多個,只是不檢查類型。

binder初始化

我們了解ioctl之后就來看看Binder設備是怎么初始化的,這里介紹的是Binder設備,并不是Binder設備驅動程序,Binder驅動程序是misc設備驅動,要想了解Binder驅動程序的內容,請點擊下面鏈接。

gityuan.com/2015/11/01/…

我們的系統服務創建的過程中,要創建打開Binder設備,下面是具體過程。 我們先來介紹下frameworks/native/libs/binder/ProcessState.cpp,ProcessState用來儲存當前進程的各種信息,系統服務啟動時會創建當前進程的ProcessState單例對象。

ProcessState::ProcessState() //打開binder : mDriverFD(open_driver()) // //映射內存的起始地址 , mVMStart(MAP_FAILED) , mThreadCountLock(PTHREAD_MUTEX_INITIALIZER) , mThreadCountDecrement(PTHREAD_COND_INITIALIZER) , mExecutingThreadsCount(0) , mMaxThreads(DEFAULT_MAX_BINDER_THREADS) , mStarvationStartTimeMs(0) , mManagesContexts(false) , mBinderContextCheckFunc(NULL) , mBinderContextUserData(NULL) , mThreadPoolStarted(false) , mThreadPoolSeq(1){ if (mDriverFD >= 0) {//分配虛擬地址空間,完成數據wirte/read,內存的memcpy等操作就相當于write/read(mDriverFD)mVMStart = mmap(0, BINDER_VM_SIZE, PROT_READ, MAP_PRIVATE | MAP_NORESERVE, mDriverFD, 0);if (mVMStart == MAP_FAILED) { close(mDriverFD); mDriverFD = -1;} }}

對于一個不懂C++的我,看起來其實挺難受的,但是這段代碼很重要,還是要看懂的。 其實我們只需要關注這幾行重要代碼 open_driver() 下面會講 mVMStart = mmap(0, BINDER_VM_SIZE, PROT_READ, MAP_PRIVATE | MAP_NORESERVE, mDriverFD, 0) 分配虛擬內存映射 我們先來看open_driver函數

static int open_driver(){ int fd = open('/dev/binder', O_RDWR | O_CLOEXEC); //打開 /dev/binder if (fd >= 0) {int vers = 0;//通過ioctl通知binder驅動binder版本status_t result = ioctl(fd, BINDER_VERSION, &vers);if (result == -1) { ALOGE('Binder ioctl to obtain version failed: %s', strerror(errno)); close(fd); fd = -1;}if (result != 0 || vers != BINDER_CURRENT_PROTOCOL_VERSION) { ALOGE('Binder driver protocol does not match user space protocol!'); close(fd); fd = -1;}//設置當前fd最多支持DEFAULT_MAX_BINDER_THREADS線程數量size_t maxThreads = DEFAULT_MAX_BINDER_THREADS;result = ioctl(fd, BINDER_SET_MAX_THREADS, &maxThreads);if (result == -1) { ALOGE('Binder ioctl to set max threads failed: %s', strerror(errno));} } else {ALOGW('Opening ’/dev/binder’ failed: %sn', strerror(errno)); } return fd;}

首先執行int fd = open('/dev/binder', O_RDWR | O_CLOEXEC); 獲取到了驅動文件的文件描述符。 文件打開成功之后,使用ioctl查詢了版本號,并設置了最大的連接線程數。 然后調用mVMStart = mmap(0, BINDER_VM_SIZE, PROT_READ, MAP_PRIVATE | MAP_NORESERVE, mDriverFD, 0)把/dev/binder文件映射到了進程虛擬內存空間,這里我們還要了解下linux的mmap函數。

mmap參考自:blog.itpub.net/7728585/vie…

在LINUX中我們可以使用mmap用來在進程虛擬地址空間中分配創建一片虛擬內存地址映射

Android Binder的原理與使用

我們可以在當前進程的虛擬內存中獲得一塊映射區域,我們直接操作映射區域就可以間接操作內核中的文件。 我們使用mmap的目的是創建共享文件映射

Android Binder的原理與使用

進程都有一份文件映射,并且都指向同一個文件,這樣就實現了共享內存,Binder就是利用這種共享內存方式去進行數據的交互。每個進程都會保留一份dev/binder設備的映射區域,這樣我們利用Binder,數據經過一次拷貝就可以實現跨進程,Linux的管道機制則需要四次拷貝

總結

1、介紹了Binder在Android中的使用方式 2、Binder機制原理,用戶進程隔離,借助內核空間進行IPC 3、使用ioctl系統調用函數,調用Binder設備驅動程序,完成IPC調用 4、dev/binder是Binder機制中的虛擬設備,利用Binder驅動可以操作該設備(數據交互) 5、mmap指令可以創建進程虛擬內存映射空間,映射dev/binder文件,實現共享內存,Binder一次拷貝原理

以上就是Android Binder的原理與使用的詳細內容,更多關于Android Binder的資料請關注好吧啦網其它相關文章!

標簽: Android
相關文章:
主站蜘蛛池模板: 亚洲国产天堂久久综合 | 久久草在线视频 | а_天堂中文最新版地址 | www.久久.com | 第一色站| 国产精品久久久久久一区二区三区 | av久久| 久久久免费视频播放 | 国产一区二区黑人欧美xxxx | 国产91麻豆视频 | 91精品国产91久久综合桃花 | 国产亲子乱弄免费视频 | 精品国产伦一区二区三区观看说明 | 成人在线亚洲 | 男人的天堂在线视频 | 中文字幕在线不卡 | 99精品欧美一区二区三区 | 夜夜操天天干, | 人人干人人干人人 | 99精品网 | 色吊丝在线永久观看最新版本 | 成人免费在线看片 | 君岛美绪一区二区三区 | 日韩欧美国产精品一区二区三区 | 一区二区免费播放 | 亚洲精品乱码 | 91在线免费观看 | 国产精品成人免费视频 | 日本成人在线视频网站 | 国产精品一区二区三区av | 亚洲一区二区三区四区五区中文 | 国产综合视频在线观看 | 亚洲国产精品久久 | 亚洲二区在线 | 91在线视频播放 | 免费在线日韩 | 伊人久久艹 | 最新中文字幕在线 | 欧美精品成人 | 午夜精品一区二区三区在线视频 | 精品免费一区二区 | 91精品国产一区二区 | 久草青青 | 一区二区三区视频播放 | 亚洲欧美日韩在线一区 | 一区二区在线免费观看 | 国产美女av | 国产色婷婷精品综合在线播放 | va在线| a在线播放 | 综合久久综合久久 | 97久久精品 | 97色综合 | 欧美精品一区二区三区蜜桃视频 | 亚洲性网 | 免费毛片网| 日韩成人高清电影 | 亚洲视频中文字幕 | 久久久久久成人 | 99久久婷婷国产综合精品电影 | 国产精品99久久 | 男女小网站 | www.视频在线观看 | 91精品久久久久久久 | 中文字幕一区二区三区乱码在线 | 中文字幕亚洲欧美日韩在线不卡 | 国产美女精品 | 免费观看视频毛片 | 天堂亚洲 | 亚洲欧美在线一区 | 欧美日韩中文 | 在线播放一级片 | 国产精品久久久久久久久小说 | 午夜国产精品视频 | 亚洲毛片网站 | 成人免费小视频 | 综合一区 | 国产精品1区2区3区 午夜视频网站 | 国产99久久精品 | 亚洲精品一区二区三区在线观看 | 久久精品亚洲一区二区 | 国产精选视频 | 欧美综合视频在线观看 | 亚洲视频一区二区三区 | 国产精品 日韩 | 男女羞羞视频在线观看 | 精品一区二区电影 | 免费v片| 91亚洲日本aⅴ精品一区二区 | 91综合网| 精品一区二区三区三区 | 97av在线 | 国产精品欧美一区二区三区 | 色噜噜视频在线观看 | 成人一区二区三区在线观看 | 三级网址日本 | www.亚洲| 亚洲色图88| 男人av网 | 中文字幕亚洲在线 | 日韩精品视频在线播放 | 奇米影视奇米色777欧美 | 午夜国产精品视频 | 日韩精品一区二区三区第95 | 99精品网| 久久精品国产亚洲一区二区三区 | 成人精品一区二区三区中文字幕 | 国产在线专区 | 国产精品人人做人人爽 | 综合久久99 | 亚洲视频在线免费观看 | 国产一区二区精品在线 | 国产精品一区二区在线观看 | 国产一区二区三区在线 | 国产精品久久久久久久久久东京 | 精品久久久久久 | 久久国产视屏 | 龙珠z在线观看 | 中文字幕精品一区 | 天天看天天爽 | 国产欧美精品一区二区三区 | 日韩天堂| 国产亚洲欧美精品永久 | 久久国产经典视频 | 久久久久高清 | 日本视频免费高清一本18 | 欧美狠狠操 | 午夜精品久久久久久久久久久久久 | 久久免费电影 | 日韩一区二区三区精品 | 97精品久久 | 日韩在线视频观看 | 精品久久久久久久久久久 | 久久精品二区亚洲w码 | 久久精品欧美一区二区三区不卡 | 91成人在线| 欧美午夜影院 | 欧美日韩国产一区二区在线观看 | 午夜网址 | 欧美一区二区三区精品免费 | 色狠狠一区 | 91嫩草在线 | 亚洲蜜臀av乱码久久精品蜜桃 | 久草在线视频免费播放 | 一区二区三区国产视频 | 伊人久久精品久久亚洲一区 | 在线观看视频一区二区 | 成人精品在线 | 97超碰在线播放 | 国产精品一区av | 国产精品久久电影观看 | 在线观看黄免费 | 日韩在线免费 | 国产精品地址 | 欧美a在线 | 欧美在线观看网站 | 久久国产一区 | 亚洲福利电影网 | 五月天中文字幕 | 密色视频 | 激情国产 | 3bmm在线观看视频免费 | 亚洲精品久久 | 在线亚洲精品 | 亚洲电影一区二区 | 国产依人在线 | 国产精品99精品久久免费 | 国产99久久久久久免费看农村 | 日韩免费网站 | 午夜精品一区二区三区在线视频 | 日本一区二区三区四区不卡视频 | 欧美电影一区 | 日韩大尺度在线观看 | 狠狠做深爱婷婷综合一区 | 免费观看a视频 | 欧美激情精品久久久久 | 欧美日韩一二三区 | 国产精品久久久久aaaa九色 | 天天久久婷婷 | 久草.com | 国产精品久久一区二区三区 | 激情欧美日韩一区二区 | 亚洲精品视频在线 | 国产精品欧美久久久久一区二区 | 亚洲欧洲精品视频 | 91久久精品 | 亚洲一区视频 | 午夜午夜精品一区二区三区文 | 欧美综合精品 | 特级淫片裸体免费看 | 成人午夜精品久久久久久久3d | а天堂中文最新一区二区三区 | 毛片入口 | 成人午夜免费网站 | 特级毛片在线 | 午夜影院网站 | 国产伦精品一区二区三区四区视频 | 国产高清在线精品一区二区三区 | 亚洲国产精品麻豆 | 久久精品店 | 欧美视频在线播放 | 一区二区三区欧美在线 | 日韩在线播放一区二区三区 | 九九九久久久 | 99精品欧美一区二区三区综合在线 | 国产精品成人在线观看 | 久久精品高清视频 | 欧美a在线看 | 99久热在线精品视频观看 | 91亚洲国产成人久久精品网站 | 午夜精| 黄色av网站在线免费观看 | 青草视频在线免费观看 | 精品网站999www | 中文字幕国产视频 | 亚洲一区二区三区免费观看 | 欧美hdfree性xxxx | 亚洲福利 | 国产精品久久 | 成人国产精品色哟哟 | 色爱av| 91人人| 91亚洲免费视频 | 亚洲视频 欧美视频 | 亚洲精品免费在线观看 | 午夜欧美 | 午夜伦理影院 | 日韩综合一区 | 乱轮一区| 美女天堂 | 成a人片在线观看 | 国产一区二区三区四区 | 国产高清第一页 | 吊视频一区二区三区 | 色婷婷亚洲一区二区三区 | 午夜免费剧场 | 国产精品a一区二区三区网址 | 在线激情视频 | 91九色视频pron| 亚洲综合无码一区二区 | 亚洲精品视频在线 | 国产成人免费在线 | 一区二区三区国产 | 日韩视频在线观看 | 亚洲中午字幕 | 天天操天天操 | 精品国产不卡一区二区三区 | 9999久久久久| 欧美精品一区二区视频 | 久久精选视频 | 亚洲欧洲精品成人久久奇米网 | 99re在线视频精品 | www.成人国产 | 久久国产一区二区三区 | 国产探花在线看 | 国产在线视频a | 精品国产一区二区三区久久影院 | 久久久蜜臀| 一区二区不卡 | 精品国产乱码久久久久久88av | 日韩三区在线 | 国产一区 | 国产精品久久久久久久久久久久久久 | 日韩中文字幕免费视频 | 精品99在线 | 欧美日韩国产一区二区 | 欧美日韩久久精品 | 日韩一区二区三区在线 | 亚洲精品久久久久avwww潮水 | 欧美精品99 | 免费二区 | 综合久久精品 | 欧美11一13sex性hd | 欧美在线亚洲 | 激情婷婷综合 | 国产成人激情 | 亚洲人免费视频 | 国产亚洲一区二区三区在线观看 | 久久韩剧网 | 综合色九九 | 日韩在线精品视频 | 久久小草 | 毛片网在线观看 | 国产 一区 | 日韩电影中文字幕 | 精品一区二区三区四区视频 | 色性网| 亚洲精品久久久久久久久久久 | 色偷偷噜噜噜亚洲男人 | 免费黄色在线视频 | 久久伊人一区 | 亚洲欧美视频 | 亚洲精品一区二区网址 | 在线中文视频 | 亚洲综合精品视频 | 午夜视频在线播放 | 日日操天天操 | 国产性×xxx盗摄xxxx | 插插射啊爱视频日a级 | 一区二区中文 | 国产伦精品久久久一区二区三区 | 日韩精品在线一区 | 偷拍做爰吃奶视频免费看 | 电影91久久久 | 91精品国产综合久久久久久丝袜 | 色婷婷综合久久久中文字幕 | 国产二区在线播放 | 日韩一区二区在线视频 | 亚洲综合区 | 亚洲综合区 | 久久精选视频 | 天天射影院 | 91在线一区二区 | 久久久久久一区二区 | 日韩综合在线 | 中文字幕在线免费观看 | 久久精品这里热有精品 | 欧美一级在线观看 | 欧美嘿咻| 91精品国产99久久久 | 色婷婷综合久久久中字幕精品久久 | 欧美一区二区三区精品免费 | 成人在线高清视频 | 黄色成人在线 | 天堂成人国产精品一区 | 久久精品国产久精国产 | 2018天天操夜夜操 | 亚洲欧洲综合av | 精品入口麻豆88视频 | www.日韩大片 | 激情视频在线观看 | 三级黄色片在线播放 | 日韩爱爱网 | 日韩中文字幕无码一区二区三区 | 午夜在线视频 | 国产福利91精品一区二区 | 亚洲精品日韩综合观看成人91 | 欧美日韩福利 | 成人国产在线 | 久久99精品久久久噜噜最新章节 | 中文在线资源 | 国产精品免费一区 | 久久久久亚洲精品 | 毛片激情永久免费 | 成人一区二区在线 | 亚洲免费精品网站 | 欧美精品影院 | 久草成人| 99久久99久久精品国产片果冻 | 91福利网址 | 精品美女久久久 | 中文字幕一区二区三区日韩精品 | 奇米二区 | 黄色小视频免费观看 | 欧美日韩亚洲综合 | 亚洲免费观看视频 | 亚洲高清在线 | 在线视频中文字幕 | 欧美激情精品久久久久久 | 99re在线 | 成人国产综合 | 一区二区三区四区在线 | 91久久久久久久久久久久久 | 国产精品人人做人人爽 | 亚洲aⅴ天堂av在线电影软件 | 免费黄色片一区二区 | 日本高清无卡码一区二区久久 | 日韩精品免费观看 | 久久久av| 久久一日本道色综合久久 | 国产成人免费视频网站高清观看视频 | 精品国产一区二区在线 | 免费观看黄a一级视频 | 免费一区二区 | 性视频网站免费 | 中文字幕在线观看 | h片在线 | 日韩福利视频网 | 欧美在线视频一区二区 | 国产四区| 91精品在线观看入口 | 97久久久 | 国产高清自拍 | 免费av一区二区三区 | jizz18毛片 | 国产精品自产拍在线观看桃花 | 大象一区| 国产综合久久久久久鬼色 | 午夜精品久久久久久久久久久久久 | 成人免费在线视频 | 久久免费精品 | 五月天婷婷国产精品 | 久久久国产精品 | 日本精品在线 | 色婷婷国产精品综合在线观看 | 91网址| 国产在线观看91一区二区三区 | 欧美二区在线 | 国产精品国产 | 国产精品一区二区三区免费 | 午夜影视 | 中文字幕精品三区 | 一区二区三区高清不卡 | 久久久99国产精品免费 | 久久综合成人精品亚洲另类欧美 | 国产综合久久久久久鬼色 | 自拍偷拍视频网站 | 国产成人综合一区二区三区 | 欧美在线高清 | 成人羞羞在线观看网站 | 久久伊人av | 国产精品久久久久久久电影 | 免费的黄色毛片 | 日韩在线一区二区三区 | 国产免费一区二区三区 | 日韩一区二区三区在线观看 | 日韩国产精品一区二区三区 | 国产中文字幕一区 | 国产精品久久久久久久久免费桃花 | 欧美日韩一区二区三区在线观看 | 一区二区三区免费 | 久久av免费 | 欧美一区二区三区男人的天堂 | 国产精品久久久久久久免费大片 | 成人观看免费视频 | 欧美激情在线免费观看 | 国产精品永久 | 国产日韩91 | 久久99精品久久久久子伦 | 69久久久 | 在线精品观看 | 国产精品一区二区不卡 | 日韩在线中文字幕 | 国产精品久久久久久吹潮 | 欧美1区| 在线欧美亚洲 | 欧美亚洲激情 | 精品国产一区二区国模嫣然 | 欧美日本精品 | 国产视频观看 | 一区二区高清 | 午夜视| 断背山在线 | 日韩一级免费在线观看 | 激情网站免费观看 | 欧美一区二区三区免费 | 91九色视频 | 日韩在线视频观看 | av在线成人 | 一本岛在线视频 | 国产一区二区在线看 | 欧美一级视频 | 91九色porny首页最多播放 | 日韩精品在线播放 | 国产在线中文字幕 | 日韩喷潮 | 日本在线观看视频网站 | 日本精品免费观看 | 国产欧美综合一区二区三区 | 欧美日韩综合精品 | 亚洲视频777| 中文字幕一区在线观看 | 亚洲一区二区av | 久久久久久久国产 | 色精品视频 | 亚洲嫩草| 欧美精品99 | 中文字幕一区二区三区在线视频 | 成人免费视频在线观看 | 日韩在线观看视频一区 | 亚洲a精品| 久久精品亚洲 | 国产成人在线一区二区 | 成人h动漫精品一区二区器材 | 亚洲一区二区 | a欧美| 青青久久 | 你懂的在线视频播放 | 97国产在线 | 久久久性色精品国产免费观看 | 亚洲精品久久久久久下一站 | 久久99精品国产麻豆婷婷洗澡 | 国产一区二区三区久久 | 国产96视频 | 国产一级免费网站 | 波多野结衣一二三四区 | 毛片久久久 | 婷婷视频在线 | 国产精品久久久久久久久久ktv | 欧美日韩a v| 久久99精品久久久久国产越南 | 午夜爱爱毛片xxxx视频免费看 | 国产日韩一区二区三区 | 性高湖久久久久久久久 | 天天爱爱网 | 国产美女久久久 | 久草新免费 | 国产婷婷精品av在线 | 亚洲一区二区三 | 免费看的av| 中文字幕日韩欧美一区二区三区 | 国产精品视频免费 | 日本久久久影视 | 综合一区二区三区 | 日韩欧美国产精品综合嫩v 久久久久久国产精品高清 国产目拍亚洲精品99久久精品 | 国产成人天天爽高清视频 | 日本成人中文字幕 | 伊人狠狠干 | 亚洲久草| 综合久久99 | 久久久久久成人精品 | 成人午夜免费网站 | 日韩精品一区二区三区在线播放 | 亚洲视频一区 | 国产精品久久久久久二区 | 久久这里只有精品首页 | 国产精品成人一区二区 | 成人在线看片 | 国产中文在线 | 日日干夜夜干 | 亚洲人成一区 | 久久激情视频 | 精品国产91亚洲一区二区三区www | 亚洲精品成人久久久 | 日韩av在线中文字幕 | 久久久久亚洲精品国产 | 99久久精品免费看国产一区二区三区 | 成人欧美一区二区三区色青冈 | 天天久久| 高清成人 | 岛国一区 | 中文字幕日韩欧美 | 性欧美大战久久久久久久免费观看 | 亚洲视频观看 | 欧美成人黄激情免费视频 | 日韩视频在线观看一区二区 | 日日操天天射 | 精品久久久中文字幕 | 日韩和的一区二区 | 特级黄一级播放 | 精品国产一区二区三区久久久蜜 | 国产第99页 | 久久一区| 伊人久久国产 | 一区二区中文字幕 | 精品国产乱码一区二区三区 | 在线91 | 欧美日韩视频一区二区 | 樱桃小丸子在线观看 | 免费黄色电影在线观看 | 国产亚洲一区二区三区在线观看 | 91精品国产自产精品男人的天堂 | 国产精品亚洲一区二区三区 | 亚洲一区二区三区蜜桃 | 黄色一级视频 | 国产精品1区2区在线观看 | 日本久久久久久久久久久久 | 国产精品成人一区二区三区夜夜夜 | 国产色片在线 | 久久精品国产亚洲一区二区三区 | 精品一区av | 亚洲精品一区二区三区樱花 | 久久久免费 | 在线成人免费观看www | 免费国产黄网站在线观看视频 | 999久久久国产999久久久 | 日韩精品第一页 | 精品国产影院 | 超碰人人射 | 91在线精品一区二区 | 性视频一区二区 | 国产精品久久免费观看spa | av午夜电影 | www日本在线 | 国产精品高颜值在线观看 | 欧美激情一区二区三级高清视频 | 午夜看片 | 91视频久久| 国户精品久久久久久久久久久不卡 | 久久国产精品视频 | 日韩国产欧美一区 | 91精品国产综合久久久蜜臀粉嫩 | 成人免费在线视频观看 | 在线观看一区二区三区四区 | 久久国产精品无码网站 | 毛片久久 | 在线视频成人 | 嫩草视频在线播放 | 天天久久 | 日本久久国产 | 欧美久久久网站 | 久久99精品久久久久久 | 神马久久久久久久 | 久久久久久91亚洲精品中文字幕 | 久久99精品国产麻豆婷婷洗澡 | 热久久这里只有精品 | 伊人av超碰久久久麻豆 | 国产在线视频xxx | 久久99视频这里只有精品 | 久久先锋 | 粉嫩视频在线观看 | 亚洲不卡视频 | 蜜臀久久99精品久久久无需会员 | 久久久久久久久久久九 | 欧美黄色一级毛片 | av在线网址观看 | 国产午夜久久 | 亚洲精品久久久久久久久久久 | www.国产.com | 黄色片网站在线观看 | 亚洲网站在线 | 精品亚洲国产成av人片传媒 | 亚洲精品国产电影 | 日韩精品视频在线观看免费 | www..99re|