日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android Binder 之 ServiceManager (基于android 12.0/S)

發布時間:2024/3/13 Android 48 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android Binder 之 ServiceManager (基于android 12.0/S) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

Binder 原理整理:

因為Linux中的進程的用戶空間是不共享的,內核空間是共享的,所以IPC通信是兩個用戶空間(APP 進程)通過共享的內核空間(Binder驅動)進行數據交互。

Binder 整體框架:

?Binder 通信框架:

ServiceManager :

ServiceManager 可執行文件的生成:

ServiceManager 在android系統中是一個可執行文件,位于/system/bin/servicemanager下面

?Servicemanager 是在init.rc中啟動的

在android 10.0.0.R47 及以前 servicemanager是由以下目錄結構編譯生成的,

?在android 10.0.0.R47 及以前 控制編譯的相關bp文件:

在android 11.0.0_r21后面原先的service_manager.c變成了ServiceManager.cpp,binder.c變成了main.cpp,同時添加了Access.cpp和Access.h,bctest 變成了 test_sm。

Android 11.0.0_r21 以后的bp如圖,先是將ServiceManager.cpp和Access.cpp一起生成了servicemanager_defaults,然后通過servicemanager_defaults編譯生成可運行的servicemanager。

?

再簡單看下目前android 12中的代碼目錄結構和 android 13的代碼結構:

?

Android T(13.0)中添加了servicemanager.microdroid.rc 和servicemanager.recovery.rc 兩個rc文件。

ServiceManager的代碼分析:

總入口:

Android S 中將android 10.0.0.R47 及以前 在service_manager.c中的 main 方法提取到了main.cpp中。main.cpp中除了main 方法外還額外有ClientCallbackCallback和BinderCallback兩個callback.

int main(int argc, char** argv) { if (argc > 2) {LOG(FATAL) << "usage: " << argv[0] << " [binder driver]";}const char* driver = argc == 2 ? argv[1] : "/dev/binder";//第二個參數可以缺省sp<ProcessState> ps = ProcessState::initWithDriver(driver);//打開binder驅動ps->setThreadPoolMaxThreadCount(0);ps->setCallRestriction(ProcessState::CallRestriction::FATAL_IF_NOT_ONEWAY);// 實例化ServiceManagersp<ServiceManager> manager = sp<ServiceManager>::make(std::make_unique<Access>());// 將自身注冊到ServiceManager當中if (!manager->addService("manager", manager, false /*allowIsolated*/, IServiceManager::DUMP_FLAG_PRIORITY_DEFAULT).isOk()) {LOG(ERROR) << "Could not self register servicemanager";}// 將ServiceManager設置給IPCThreadState的全局變量IPCThreadState::self()->setTheContextObject(manager);ps->becomeContextManager();//注冊成為binder服務的大管家// 準備Loopersp<Looper> looper = Looper::prepare(false /*allowNonCallbacks*/);//給Looper設置callback BinderCallback::setupTo(looper);ClientCallbackCallback::setupTo(looper, manager);//進入無限循環,處理client端發來的請求while(true) {looper->pollAll(-1);}// should not be reachedreturn EXIT_FAILURE; }

?下圖為android 10.0.0.R47 及以前 在service_manager.c中的 main 方法(因為頁面截圖空間限制沒有截全 可以自行查看 http://aospxref.com/android-10.0.0_r47/xref/frameworks/native/cmds/servicemanager/service_manager.c#382),相關代碼講解可以參考http://gityuan.com/2015/11/07/binder-start-sm/

其中main方法中主要干了四件事:

1)初始化binder驅動

2)將自身以“manager” 注冊到servicemanager中

3)注冊成為binder服務的大管家

4) 給Looper設置callback,進入無限循環,處理client端發來的請求

這里面著重講后三個代碼塊

1)第一個代碼塊中,android 10.0.0.R47 之前是通過binder_open 直接操作binder驅動,沒有借助libbinder,Android 11.0.0_r21 以后是通過initWithDriver 對于binder進行操作的,在編譯servicemanager的時候,添加了libbinder的庫依賴進去。

2)第二個代碼塊,將自身以“manager” 注冊到servicemanager中:

Status ServiceManager::addService(const std::string& name, const sp<IBinder>& binder, bool allowIsolated, int32_t dumpPriority) { auto ctx = mAccess->getCallingContext();//獲取到調用的Context// apps cannot add services (AID_APP =10000)if (multiuser_get_app_id(ctx.uid) >= AID_APP) {return Status::fromExceptionCode(Status::EX_SECURITY);}if (!mAccess->canAdd(ctx, name)) {return Status::fromExceptionCode(Status::EX_SECURITY);}if (binder == nullptr) {return Status::fromExceptionCode(Status::EX_ILLEGAL_ARGUMENT);}if (!isValidServiceName(name)) {LOG(ERROR) << "Invalid service name: " << name;return Status::fromExceptionCode(Status::EX_ILLEGAL_ARGUMENT);}#ifndef VENDORSERVICEMANAGERif (!meetsDeclarationRequirements(binder, name)) {// already loggedreturn Status::fromExceptionCode(Status::EX_ILLEGAL_ARGUMENT);} #endif // !VENDORSERVICEMANAGER// implicitly unlinked when the binder is removed if (binder->remoteBinder() != nullptr &&binder->linkToDeath(sp<ServiceManager>::fromExisting(this)) != OK) {LOG(ERROR) << "Could not linkToDeath when adding " << name;return Status::fromExceptionCode(Status::EX_ILLEGAL_STATE);}//以上代碼多是異常情況處理 // Overwrite the old service if it exists//將service的相關信息寫入到 servicemanager的 map中mNameToService[name] = Service {.binder = binder,.allowIsolated = allowIsolated,.dumpPriority = dumpPriority,.debugPid = ctx.debugPid,};auto it = mNameToRegistrationCallback.find(name);if (it != mNameToRegistrationCallback.end()) {for (const sp<IServiceCallback>& cb : it->second) {mNameToService[name].guaranteeClient = true;// permission checked in registerForNotificationscb->onRegistration(name, binder);}}return Status::ok(); }

上面的addService 涉及到 Binder的報錯類型枚舉類:

?servicemanager中維護注冊服務的map:

?分解看service類和ServiceMap:

?這里可以看到servicemanager是用map維護注冊的服務的,android 10.0.0.R47 及以前是通過鏈表進行維護的。這里面猜測數據結構的變化是隨著手機代碼的內存增大和性能指標的增強,鏈表省空間但是查詢較慢的特性已經不能滿足需求,于是改用了查詢更快的 map進行存儲。下圖是 android 10.0.0.R47 的注冊方法:?

3)第三個代碼塊,servicemanager 成為binder服務的大管家。此處通過ioctl往binder驅動發了#define BINDER_SET_CONTEXT_MGR_EXT _IOW('b', 13, struct flat_binder_object) 的命令,如果不好用則按照android 10.0.0.R47的方式發 #define BINDER_SET_CONTEXT_MGR _IOW('b', 7, __s32)。后續流程的拆解歡迎大家幫忙補充下。

?4)第四個代碼塊,給Looper設置callback,進入無限循環,處理client端發來的請求

給Looper 設置了BinderCallback 和 ClientCallbackCallback,兩個callback 都是Loopercallback的子類。

?

?

class BinderCallback : public LooperCallback { public:static sp<BinderCallback> setupTo(const sp<Looper>& looper) { // 實例化BinderCallback sp<BinderCallback> cb = sp<BinderCallback>::make();int binder_fd = -1;//通過IPCThreadState獲取binder_fd,這里面的IPCThreadState待補充IPCThreadState::self()->setupPolling(&binder_fd);LOG_ALWAYS_FATAL_IF(binder_fd < 0, "Failed to setupPolling: %d", binder_fd);//添加監聽目標int ret = looper->addFd(binder_fd,Looper::POLL_CALLBACK,Looper::EVENT_INPUT,cb,nullptr /*data*/);LOG_ALWAYS_FATAL_IF(ret != 1, "Failed to add binder FD to Looper");return cb;}int handleEvent(int /* fd */, int /* events */, void* /* data */) override { //處理回調IPCThreadState::self()->handlePolledCommands();return 1; // Continue receiving callbacks.} };

?

Looper會監聽ServiceManager 進程中打開的binder_fd,有消息上來了會調用handlePolledCommands處理。

核心是getAndExecuteCommand方法:?

status_t IPCThreadState::getAndExecuteCommand(){status_t result;int32_t cmd;//從binder driver獲取mIn數據result = talkWithDriver();if (result >= NO_ERROR) {size_t IN = mIn.dataAvail();if (IN < sizeof(int32_t)) return result;cmd = mIn.readInt32();IF_LOG_COMMANDS() {alog << "Processing top-level Command: "<< getReturnString(cmd) << endl;}pthread_mutex_lock(&mProcess->mThreadCountLock);mProcess->mExecutingThreadsCount++;if (mProcess->mExecutingThreadsCount >= mProcess->mMaxThreads &&mProcess->mStarvationStartTimeMs == 0) {mProcess->mStarvationStartTimeMs = uptimeMillis();}pthread_mutex_unlock(&mProcess->mThreadCountLock);// 解析出對應的cmd,執行cmdresult = executeCommand(cmd);pthread_mutex_lock(&mProcess->mThreadCountLock);mProcess->mExecutingThreadsCount--;if (mProcess->mExecutingThreadsCount < mProcess->mMaxThreads &&mProcess->mStarvationStartTimeMs != 0) {int64_t starvationTimeMs = uptimeMillis() - mProcess->mStarvationStartTimeMs;if (starvationTimeMs > 100) {ALOGE("binder thread pool (%zu threads) starved for %" PRId64 " ms",mProcess->mMaxThreads, starvationTimeMs);}mProcess->mStarvationStartTimeMs = 0;}// Cond broadcast can be expensive, so don't send it every time a binder// call is processed. b/168806193if (mProcess->mWaitingForThreads > 0) {pthread_cond_broadcast(&mProcess->mThreadCountDecrement);}pthread_mutex_unlock(&mProcess->mThreadCountLock);}return result; } status_t IPCThreadState::executeCommand(int32_t cmd){BBinder* obj;RefBase::weakref_type* refs;status_t result = NO_ERROR;switch ((uint32_t)cmd) {case BR_ERROR:result = mIn.readInt32();break;case BR_OK:break;case BR_ACQUIRE:refs = (RefBase::weakref_type*)mIn.readPointer();obj = (BBinder*)mIn.readPointer();ALOG_ASSERT(refs->refBase() == obj,"BR_ACQUIRE: object %p does not match cookie %p (expected %p)",refs, obj, refs->refBase());obj->incStrong(mProcess.get());IF_LOG_REMOTEREFS() {LOG_REMOTEREFS("BR_ACQUIRE from driver on %p", obj);obj->printRefs();}mOut.writeInt32(BC_ACQUIRE_DONE);mOut.writePointer((uintptr_t)refs);mOut.writePointer((uintptr_t)obj);break;case BR_RELEASE:refs = (RefBase::weakref_type*)mIn.readPointer();obj = (BBinder*)mIn.readPointer();ALOG_ASSERT(refs->refBase() == obj,"BR_RELEASE: object %p does not match cookie %p (expected %p)",refs, obj, refs->refBase());IF_LOG_REMOTEREFS() {LOG_REMOTEREFS("BR_RELEASE from driver on %p", obj);obj->printRefs();}mPendingStrongDerefs.push(obj);break;case BR_INCREFS:refs = (RefBase::weakref_type*)mIn.readPointer();obj = (BBinder*)mIn.readPointer();refs->incWeak(mProcess.get());mOut.writeInt32(BC_INCREFS_DONE);mOut.writePointer((uintptr_t)refs);mOut.writePointer((uintptr_t)obj);break;case BR_DECREFS:refs = (RefBase::weakref_type*)mIn.readPointer();obj = (BBinder*)mIn.readPointer();// NOTE: This assertion is not valid, because the object may no// longer exist (thus the (BBinder*)cast above resulting in a different// memory address).//ALOG_ASSERT(refs->refBase() == obj,// "BR_DECREFS: object %p does not match cookie %p (expected %p)",// refs, obj, refs->refBase());mPendingWeakDerefs.push(refs);break;case BR_ATTEMPT_ACQUIRE:refs = (RefBase::weakref_type*)mIn.readPointer();obj = (BBinder*)mIn.readPointer();{const bool success = refs->attemptIncStrong(mProcess.get());ALOG_ASSERT(success && refs->refBase() == obj,"BR_ATTEMPT_ACQUIRE: object %p does not match cookie %p (expected %p)",refs, obj, refs->refBase());mOut.writeInt32(BC_ACQUIRE_RESULT);mOut.writeInt32((int32_t)success);}break;case BR_TRANSACTION_SEC_CTX:case BR_TRANSACTION:{//讀取mIn中的數據到一個binder_transaction_data中binder_transaction_data_secctx tr_secctx;binder_transaction_data& tr = tr_secctx.transaction_data;if (cmd == (int) BR_TRANSACTION_SEC_CTX) {result = mIn.read(&tr_secctx, sizeof(tr_secctx));} else {result = mIn.read(&tr, sizeof(tr));tr_secctx.secctx = 0;}ALOG_ASSERT(result == NO_ERROR,"Not enough command data for brTRANSACTION");if (result != NO_ERROR) break;Parcel buffer;buffer.ipcSetDataReference(reinterpret_cast<const uint8_t*>(tr.data.ptr.buffer),tr.data_size,reinterpret_cast<const binder_size_t*>(tr.data.ptr.offsets),tr.offsets_size/sizeof(binder_size_t), freeBuffer);const void* origServingStackPointer = mServingStackPointer;mServingStackPointer = &origServingStackPointer; // anything on the stackconst pid_t origPid = mCallingPid;const char* origSid = mCallingSid;const uid_t origUid = mCallingUid;const int32_t origStrictModePolicy = mStrictModePolicy;const int32_t origTransactionBinderFlags = mLastTransactionBinderFlags;const int32_t origWorkSource = mWorkSource;const bool origPropagateWorkSet = mPropagateWorkSource;// Calling work source will be set by Parcel#enforceInterface. Parcel#enforceInterface// is only guaranteed to be called for AIDL-generated stubs so we reset the work source// here to never propagate it.clearCallingWorkSource();clearPropagateWorkSource();mCallingPid = tr.sender_pid;mCallingSid = reinterpret_cast<const char*>(tr_secctx.secctx);mCallingUid = tr.sender_euid;mLastTransactionBinderFlags = tr.flags;// ALOGI(">>>> TRANSACT from pid %d sid %s uid %d\n", mCallingPid,// (mCallingSid ? mCallingSid : "<N/A>"), mCallingUid);Parcel reply;status_t error;IF_LOG_TRANSACTIONS() {TextOutput::Bundle _b(alog);alog << "BR_TRANSACTION thr " << (void*)pthread_self()<< " / obj " << tr.target.ptr << " / code "<< TypeCode(tr.code) << ": " << indent << buffer<< dedent << endl<< "Data addr = "<< reinterpret_cast<const uint8_t*>(tr.data.ptr.buffer)<< ", offsets addr="<< reinterpret_cast<const size_t*>(tr.data.ptr.offsets) << endl;}if (tr.target.ptr) {// We only have a weak reference on the target object, so we must first try to// safely acquire a strong reference before doing anything else with it.if (reinterpret_cast<RefBase::weakref_type*>(tr.target.ptr)->attemptIncStrong(this)) {error = reinterpret_cast<BBinder*>(tr.cookie)->transact(tr.code, buffer,&reply, tr.flags);reinterpret_cast<BBinder*>(tr.cookie)->decStrong(this);} else {error = UNKNOWN_TRANSACTION;}} else {//調用BBinder的transact方法error = the_context_object->transact(tr.code, buffer, &reply, tr.flags);}//打開該處log可以用來調試//ALOGI("<<<< TRANSACT from pid %d restore pid %d sid %s uid %d\n",// mCallingPid, origPid, (origSid ? origSid : "<N/A>"), origUid);if ((tr.flags & TF_ONE_WAY) == 0) {LOG_ONEWAY("Sending reply to %d!", mCallingPid);if (error < NO_ERROR) reply.setError(error);constexpr uint32_t kForwardReplyFlags = TF_CLEAR_BUF;//將返回的結果重新發給bindersendReply(reply, (tr.flags & kForwardReplyFlags));} else {if (error != OK) {alog << "oneway function results for code " << tr.code<< " on binder at "<< reinterpret_cast<void*>(tr.target.ptr)<< " will be dropped but finished with status "<< statusToString(error);// ideally we could log this even when error == OK, but it// causes too much logspam because some manually-written// interfaces have clients that call methods which always// write results, sometimes as oneway methods.if (reply.dataSize() != 0) {alog << " and reply parcel size " << reply.dataSize();}alog << endl;}LOG_ONEWAY("NOT sending reply to %d!", mCallingPid);}mServingStackPointer = origServingStackPointer;mCallingPid = origPid;mCallingSid = origSid;mCallingUid = origUid;mStrictModePolicy = origStrictModePolicy;mLastTransactionBinderFlags = origTransactionBinderFlags;mWorkSource = origWorkSource;mPropagateWorkSource = origPropagateWorkSet;IF_LOG_TRANSACTIONS() {TextOutput::Bundle _b(alog);alog << "BC_REPLY thr " << (void*)pthread_self() << " / obj "<< tr.target.ptr << ": " << indent << reply << dedent << endl;}}break;case BR_DEAD_BINDER:{BpBinder *proxy = (BpBinder*)mIn.readPointer();proxy->sendObituary();mOut.writeInt32(BC_DEAD_BINDER_DONE);mOut.writePointer((uintptr_t)proxy);} break;case BR_CLEAR_DEATH_NOTIFICATION_DONE:{BpBinder *proxy = (BpBinder*)mIn.readPointer();proxy->getWeakRefs()->decWeak(proxy);} break;case BR_FINISHED:result = TIMED_OUT;break;case BR_NOOP:break;case BR_SPAWN_LOOPER:mProcess->spawnPooledThread(false);break;default:ALOGE("*** BAD COMMAND %d received from Binder driver\n", cmd);result = UNKNOWN_ERROR;break;}if (result != NO_ERROR) {mLastError = result;}return result; }

ClientCallbackCallback:

// LooperCallback for IClientCallbackclass ClientCallbackCallback : public LooperCallback { public:static sp<ClientCallbackCallback> setupTo(const sp<Looper>& looper, const sp<ServiceManager>& manager) { sp<ClientCallbackCallback> cb = sp<ClientCallbackCallback>::make(manager);//創建一個定時器描述符timerfdint fdTimer = timerfd_create(CLOCK_MONOTONIC, 0 /*flags*/);LOG_ALWAYS_FATAL_IF(fdTimer < 0, "Failed to timerfd_create: fd: %d err: %d", fdTimer, errno);itimerspec timespec {.it_interval = {.tv_sec = 5,.tv_nsec = 0,},.it_value = {.tv_sec = 5,.tv_nsec = 0,},};//啟動所有的定時器int timeRes = timerfd_settime(fdTimer, 0 /*flags*/, &timespec, nullptr);LOG_ALWAYS_FATAL_IF(timeRes < 0, "Failed to timerfd_settime: res: %d err: %d", timeRes, errno);//以時間為描述符注冊到Looper中int addRes = looper->addFd(fdTimer,Looper::POLL_CALLBACK,Looper::EVENT_INPUT,cb,nullptr);LOG_ALWAYS_FATAL_IF(addRes != 1, "Failed to add client callback FD to Looper");return cb;}int handleEvent(int fd, int /*events*/, void* /*data*/) override { uint64_t expirations;int ret = read(fd, &expirations, sizeof(expirations));if (ret != sizeof(expirations)) {ALOGE("Read failed to callback FD: ret: %d err: %d", ret, errno);}mManager->handleClientCallbacks();return 1; // Continue receiving callbacks.} private:friend sp<ClientCallbackCallback>;ClientCallbackCallback(const sp<ServiceManager>& manager) : mManager(manager) {}sp<ServiceManager> mManager; };

當looper接收到消息時候,調用servicemanager的 handleClientCallbacks進行處理。

?主要調用handleServiceClientCallback進行處理:

ssize_t ServiceManager::handleServiceClientCallback(const std::string& serviceName, bool isCalledOnInterval) {auto serviceIt = mNameToService.find(serviceName);if (serviceIt == mNameToService.end() || mNameToClientCallback.count(serviceName) < 1) {return -1;}Service& service = serviceIt->second;ssize_t count = service.getNodeStrongRefCount();// binder driver doesn't support this featureif (count == -1) return count;bool hasClients = count > 1; // this process holds a strong countif (service.guaranteeClient) {// we have no record of this clientif (!service.hasClients && !hasClients) {sendClientCallbackNotifications(serviceName, true);}// guarantee is temporaryservice.guaranteeClient = false;}// only send notifications if this was called via the interval checking workflowif (isCalledOnInterval) {if (hasClients && !service.hasClients) {// client was retrieved in some other waysendClientCallbackNotifications(serviceName, true);}// there are no more clients, but the callback has not been called yetif (!hasClients && service.hasClients) {sendClientCallbackNotifications(serviceName, false);}}return count; }

?最后通過Looper.pollAll進入無限循環,如果Looper收到消息則觸發callback。

servicemanager的主要功能:

1)注冊服務

其中注冊服務主要是通過addService 方法實現的,在講解總入口第二個代碼塊的時候已經做過拆解,不再贅余。

2)查詢服務

sp<IBinder> ServiceManager::tryGetService(const std::string& name, bool startIfNotFound) { auto ctx = mAccess->getCallingContext();sp<IBinder> out;Service* service = nullptr;if (auto it = mNameToService.find(name); it != mNameToService.end()) {service = &(it->second);if (!service->allowIsolated) {uid_t appid = multiuser_get_app_id(ctx.uid);bool isIsolated = appid >= AID_ISOLATED_START && appid <= AID_ISOLATED_END;if (isIsolated) {return nullptr;}}//將map中的信息賦值out = service->binder;}if (!mAccess->canFind(ctx, name)) {return nullptr;}//如果找不到對應的service,則嘗試以AIDL的方式啟動該serviceif (!out && startIfNotFound) {tryStartService(name);}if (out) {// Setting this guarantee each time we hand out a binder ensures that the client-checking// loop knows about the event even if the client immediately drops the serviceservice->guaranteeClient = true;}return out; }

3)獲取servicemanager

不論是注冊服務或者查詢服務,都需要先獲得servicemanager的實例,servicemanager是通過defaultServiceManager 獲取的,

[[clang::no_destroy]] static std::once_flag gSmOnce; sp<IServiceManager> defaultServiceManager(){std::call_once(gSmOnce, []() {//AidlServiceManager?就是IServiceManagersp<AidlServiceManager> sm = nullptr;while (sm == nullptr) {sm = interface_cast<AidlServiceManager>(ProcessState::self()->getContextObject(nullptr));if (sm == nullptr) {ALOGE("Waiting 1s on context object on %s.", ProcessState::self()->getDriverName().c_str());sleep(1);}}gDefaultServiceManager = sp<ServiceManagerShim>::make(sm);});return gDefaultServiceManager; }

這里面的gSmOnce和call_once 從名字看是只調用一次的意思,這里先不求甚解。類比android 10.0之前是使用的單例模式,此處的功能應該是類似的。

如圖,AidlServiceManager 就是IServiceManager。

這里與一般的單例模式不太一樣,里面多了一層while循環,這是google在2013年1月Todd Poynor提交的修改。當嘗試創建或獲取ServiceManager時,ServiceManager可能尚未準備就緒,這時通過sleep 1秒后,循環嘗試獲取直到成功。gDefaultServiceManager的創建過程,可分解為以下3個步驟

  • ProcessState::self():用于獲取ProcessState對象(也是單例模式),每個進程有且只有一個ProcessState對象,存在則直接返回,不存在則創建;

  • getContextObject(): 用于獲取BpBinder對象,對于handle=0的BpBinder對象,存在則直接返回,不存在才創建;

  • interface_cast<AidlServiceManager>():用于獲取BpServiceManager對象;

分開講三個流程:

1)ProcessState::self() 獲取ProcessState對象

sp<ProcessState> ProcessState::init(const char *driver, bool requireDefault){[[clang::no_destroy]] static sp<ProcessState> gProcess;[[clang::no_destroy]] static std::mutex gProcessMutex;if (driver == nullptr) {std::lock_guard<std::mutex> l(gProcessMutex);return gProcess;}[[clang::no_destroy]] static std::once_flag gProcessOnce;std::call_once(gProcessOnce, [&](){if (access(driver, R_OK) == -1) {ALOGE("Binder driver %s is unavailable. Using /dev/binder instead.", driver);driver = "/dev/binder";}std::lock_guard<std::mutex> l(gProcessMutex);//ProcessState調用構造方法進行初始化gProcess = sp<ProcessState>::make(driver);});if (requireDefault) {// Detect if we are trying to initialize with a different driver, and// consider that an error. ProcessState will only be initialized once above.LOG_ALWAYS_FATAL_IF(gProcess->getDriverName() != driver,"ProcessState was already initialized with %s,"" can't initialize with %s.",gProcess->getDriverName().c_str(), driver);}return gProcess; }

ProcessState::ProcessState(const char *driver) : mDriverName(String8(driver)), mDriverFD(open_driver(driver))//打開Binder驅動, mVMStart(MAP_FAILED), mThreadCountLock(PTHREAD_MUTEX_INITIALIZER), mThreadCountDecrement(PTHREAD_COND_INITIALIZER), mExecutingThreadsCount(0), mWaitingForThreads(0), mMaxThreads(DEFAULT_MAX_BINDER_THREADS), mStarvationStartTimeMs(0), mThreadPoolStarted(false), mThreadPoolSeq(1), mCallRestriction(CallRestriction::NONE) {if (mDriverFD >= 0) {// mmap the binder, providing a chunk of virtual address space to receive transactions. //mmap binder驅動,提供一個虛擬內存空間地址用于收到事務//#define BINDER_VM_SIZE ((1 * 1024 * 1024) - sysconf(_SC_PAGE_SIZE) * 2)mVMStart = mmap(nullptr, BINDER_VM_SIZE, PROT_READ, MAP_PRIVATE | MAP_NORESERVE, mDriverFD, 0);if (mVMStart == MAP_FAILED) {// *sigh*ALOGE("Using %s failed: unable to mmap transaction memory.\n", mDriverName.c_str());close(mDriverFD);mDriverFD = -1;mDriverName.clear();}}#ifdef __ANDROID__LOG_ALWAYS_FATAL_IF(mDriverFD < 0, "Binder driver '%s' could not be opened. Terminating.", driver); #endif }

?打開binder驅動代碼塊:

static int open_driver(const char *driver){// 打開/dev/binder設備,建立與內核的Binder驅動的交互通道int fd = open(driver, O_RDWR | O_CLOEXEC);if (fd >= 0) {int vers = 0;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(%d) does not match user space protocol(%d)! ioctl() return value: %d",vers, BINDER_CURRENT_PROTOCOL_VERSION, result);close(fd);fd = -1;}size_t maxThreads = DEFAULT_MAX_BINDER_THREADS;// 通過ioctl設置binder驅動,能支持的最大線程數//#define DEFAULT_MAX_BINDER_THREADS 15 默認是15個線程result = ioctl(fd, BINDER_SET_MAX_THREADS, &maxThreads);if (result == -1) {ALOGE("Binder ioctl to set max threads failed: %s", strerror(errno));}uint32_t enable = DEFAULT_ENABLE_ONEWAY_SPAM_DETECTION;result = ioctl(fd, BINDER_ENABLE_ONEWAY_SPAM_DETECTION, &enable);if (result == -1) {ALOGD("Binder ioctl to enable oneway spam detection failed: %s", strerror(errno));}} else {ALOGW("Opening '%s' failed: %s\n", driver, strerror(errno));}return fd; }

2)getContextObject(): 獲取BpBinder對象

獲取handle=0的IBinder

sp<IBinder> ProcessState::getStrongProxyForHandle(int32_t handle){sp<IBinder> result;AutoMutex _l(mLock);//查找handle對應的資源項handle_entry* e = lookupHandleLocked(handle);if (e != nullptr) {// We need to create a new BpBinder if there isn't currently one, OR we// are unable to acquire a weak reference on this current one. The// attemptIncWeak() is safe because we know the BpBinder destructor will always// call expungeHandle(), which acquires the same lock we are holding now.// We need to do this because there is a race condition between someone// releasing a reference on this BpBinder, and a new reference on its handle// arriving from the driver.IBinder* b = e->binder;if (b == nullptr || !e->refs->attemptIncWeak(this)) {if (handle == 0) {// Special case for context manager...// The context manager is the only object for which we create// a BpBinder proxy without already holding a reference.// Perform a dummy transaction to ensure the context manager// is registered before we create the first local reference// to it (which will occur when creating the BpBinder).// If a local reference is created for the BpBinder when the// context manager is not present, the driver will fail to// provide a reference to the context manager, but the// driver API does not return status. Note that this is not race-free if the context manager// dies while this code runs. TODO: add a driver API to wait for context manager, or// stop special casing handle 0 for context manager and add// a driver API to get a handle to the context manager with// proper reference counting.IPCThreadState* ipc = IPCThreadState::self();CallRestriction originalCallRestriction = ipc->getCallRestriction();ipc->setCallRestriction(CallRestriction::NONE);Parcel data;status_t status = ipc->transact(0, IBinder::PING_TRANSACTION, data, nullptr, 0);//通過ping操作測試binder是否準備就緒ipc->setCallRestriction(originalCallRestriction);if (status == DEAD_OBJECT)return nullptr;}//當handle值所對應的IBinder不存在或弱引用無效時,則創建BpBinder對象sp<BpBinder> b = BpBinder::create(handle);e->binder = b.get();if (b) e->refs = b->getWeakRefs();result = b;} else {// This little bit of nastyness is to allow us to add a primary// reference to the remote proxy when this team doesn't have one// but another team is sending the handle to us.result.force_set(b);e->refs->decWeak(this);}}return result; }

如果handle 為0的Ibinder存在且通過Ping 測試已經準備好了,則返回該Ibinder,當handle值所對應的IBinder不存在或弱引用無效時,則創建BpBinder對象。

ProcessState::handle_entry* ProcessState::lookupHandleLocked(int32_t handle){const size_t N=mHandleToObject.size();//當handle大于mHandleToObject的長度時,進入該分支if (N <= (size_t)handle) {handle_entry e;e.binder = nullptr;e.refs = nullptr;//從mHandleToObject的第N個位置開始,插入(handle+1-N)個e到隊列中status_t err = mHandleToObject.insertAt(e, N, handle+1-N);if (err < NO_ERROR) return nullptr;}return &mHandleToObject.editItemAt(handle); }

(下面模板函數部分文案出自GitYuan,非原創)根據handle值來查找對應的handle_entry,handle_entry是一個結構體,里面記錄IBinder和weakref_type兩個指針。當handle大于mHandleToObject的Vector長度時,則向該Vector中添加(handle+1-N)個handle_entry結構體,然后再返回handle向對應位置的handle_entry結構體指針。

當handle值所對應的IBinder不存在或弱引用無效時,創建BpBinder并延長對象的生命時間,創建BpBinder對象中會將handle相對應Binder的弱引用增加1:

?3)interface_cast<AidlServiceManager>():獲取BpServiceManager對象

AidlServiceManager就是IServiceManager,所以主要拆解 interface_cast:

?(interface_cast<IServiceManager>() 等價于 IServiceManager::asInterface(),asInterface是通過模板函數來定義的,

主要由以下兩個部分構成:

① DECLARE_META_INTERFACE(IServiceManager)

② IMPLEMENT_META_INTERFACE(IServiceManager,"android.os.IServiceManager")

?對于IServiceManager來說只需要換INTERFACE=IServiceManager即可,

DECLARE_META_INTERFACE 過程主要是聲明asInterface(),getInterfaceDescriptor()方法。

IMPLEMENT_META_INTERFACE 過程:

?對于IServiceManager來說 INTERFACE=IServiceManager, NAME=”android.os.IServiceManager”,可以看到DECLARE_META_INTERFACE 中的IServiceManager::asInterface() 等價于 BpIServiceManager()::make(obj)。在這里,更確切地說應該是BpIServiceManager::make(BpBinder)。

BpIServiceManager/BpServiceManager 的構造暫時未找到,能力有限,模板函數并不是很熟悉,此處文大家可以參考GitYuan的博客http://gityuan.com/2015/11/08/binder-get-sm/ 先看下android 11.0之前的講解。后續會補上android 12部分的拆解

?總結來看,defaultServiceManager 幾近 等于BpIServiceManager::make(BpBinder),這樣就獲得到了serviceManager的proxy,類似systemServer 中調用PackageManagerService的方法要拿到PackageManager 一樣,后續就可以調用serviceManager中的addService和getService方法了。

?從網上找到的 Android 11 之前的 啟動流程圖,可以先借助理解下,后面會更新最新的圖示:

?權限控制模塊:

Access 主要是通過Selinux來進行權限控制的

1)注冊服務的時候的校驗:

由于manager在service_contexts中注冊了,所以這塊Selinux可以順利通過。?

2)在查詢服務的時候通過canFind對于權限進行校驗。?

最后和 addService一樣也會通過actionAllowedFromLookup 進行校驗。

?對外接口:

在android 11之前對外接口只有四個:

在android 12中擴充為14個:?(這里將binderDied和handleClientCallbacks計算在內了)

?相關面試題:

servicemanager映射的虛擬內存有多大?現在的答案是和普通應用一樣大:1M-2頁。

frameworks/native/libs/binder/ProcessState.cpp

#define BINDER_VM_SIZE ((1 * 1024 * 1024) - sysconf(_SC_PAGE_SIZE) * 2)

參考資料:

Android 12 系統源碼分析 | Native Binder 代碼變遷 - 秋城 - 博客園

Android 12(S) Binder(一) - 青山渺渺 - 博客園

www.jb51.net

Binder系列-開篇 - Gityuan博客 | 袁輝輝的技術博客

Binder系列3-啟動ServiceManager - Gityuan博客 | 袁輝輝的技術博客

總結

以上是生活随笔為你收集整理的Android Binder 之 ServiceManager (基于android 12.0/S)的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

av在线播放免费 | 欧美一级特黄aaaaaa大片在线观看 | 色姑娘综合天天 | zzijzzij亚洲日本少妇熟睡 | 欧洲亚洲女同hd | 久久成人国产 | 黄色av免费| 亚洲男女精品 | 999成人| 久久99精品久久久久久久久久久久 | 黄网站www| 亚洲精品xxx | 久久久综合精品 | 激情视频一区 | 黄色小视频在线观看免费 | 一区免费观看 | 久久综合激情 | 亚洲国内精品 | 四虎在线免费 | 人人搞人人干 | 久久国产电影院 | 中文字幕在线视频国产 | 天天干天天在线 | 国产福利一区二区三区视频 | 国产原创在线视频 | 久久都是精品 | 精品一区二区三区在线播放 | 韩国精品一区二区三区六区色诱 | 日韩二区三区在线观看 | 亚洲伊人第一页 | 美女网站色免费 | 欧美精品免费在线观看 | 国产成视频在线观看 | 亚洲精品视频免费看 | 奇米网在线观看 | 在线观看视频色 | 麻豆久久一区二区 | 99在线精品视频 | 一区二区不卡在线观看 | 四虎在线永久免费观看 | 欧美色综合久久 | 最新中文字幕 | 久草网站在线观看 | 免费福利在线视频 | 成人少妇影院yyyy | 午夜精品视频免费在线观看 | 97色在线观看免费视频 | 日韩精品视频在线观看免费 | 91看片在线观看 | 夜夜操天天干, | 日韩精品免费在线观看 | 欧美性爽爽 | 五月婷av| 国产高清视频免费在线观看 | 免费色av| 又长又大又黑又粗欧美 | 手机看片午夜 | 亚洲一级片免费观看 | 天天拍天天色 | 国产精品久久久久久久久久久久久 | 草久草久 | 亚洲人成在线电影 | 日韩在线免费视频观看 | 99久精品 | 在线观看香蕉视频 | 99精品热视频只有精品10 | 超碰97成人| 98超碰在线 | 99 视频 高清 | 成人午夜精品 | 区一区二区三区中文字幕 | 国产艹b视频 | 69久久99精品久久久久婷婷 | 婷婷丁香激情五月 | 天天爽天天爽夜夜爽 | 亚洲精品在线一区二区 | 国产一区二区午夜 | 偷拍区另类综合在线 | 亚洲一区动漫 | 日日夜夜天天射 | 国产毛片在线 | 九九免费在线视频 | 成人免费视频网站在线观看 | 99精品国产兔费观看久久99 | 一区在线播放 | 国产精品久久久久久久免费观看 | 久久久久女人精品毛片 | 亚洲激情婷婷 | 日韩电影中文,亚洲精品乱码 | 国产精品久久中文字幕 | 欧美激情视频在线观看免费 | 国产剧情一区二区 | 麻豆视频在线看 | 亚洲丝袜一区二区 | 中文字幕麻豆 | 日韩欧美一级二级 | 成人国产精品免费观看 | 国产一级视频在线 | 九九热在线精品 | 免费高清在线观看电视网站 | 国产视频1| 在线观看国产成人av片 | 国产一区二区三区在线免费观看 | 永久精品视频 | 在线看片成人 | 热久久精品在线 | 热99久久精品 | 久草在线最新 | 久久露脸国产精品 | 久久不卡国产精品一区二区 | 午夜免费电影院 | 成 人 免费 黄 色 视频 | 亚洲无线视频 | av免费观看高清 | 国产在线欧美在线 | 97超碰人 | 亚洲精品日韩一区二区电影 | 精品嫩模福利一区二区蜜臀 | 日日爱夜夜爱 | 香蕉视频久久 | 97精品国产97久久久久久免费 | 欧美激情一区不卡 | 日韩视频一区二区在线 | 91日韩在线视频 | bbbb操bbbb| 亚洲精品视频在线观看视频 | 午夜免费福利片 | 国产在线观看免 | 色九九在线 | 国产午夜精品一区二区三区欧美 | 亚洲精品高清一区二区三区四区 | 国产精品永久在线 | 欧美日韩在线免费观看 | av免费高清观看 | 国产午夜精品一区二区三区在线观看 | 亚洲国产网址 | 99在线视频免费观看 | 少妇搡bbbb搡bbb搡忠贞 | 欧美综合国产 | 91亚洲精品久久久 | 免费激情在线电影 | 精品视频亚洲 | 国产视频一| 97精品国产91久久久久久 | 亚洲精品视频在线播放 | 丁香5月婷婷久久 | 中文字幕 第二区 | 热re99久久精品国产66热 | 精一区二区 | 国产不卡在线观看 | 久久精品中文字幕一区二区三区 | 国产精品区免费视频 | 欧美日韩国产色综合一二三四 | 97超碰中文字幕 | 黄色片视频在线观看 | 欧美精品做受xxx性少妇 | 97免费在线观看视频 | 天天操天天摸天天射 | a爱爱视频 | 天堂网一区二区三区 | 亚洲资源在线观看 | 国产小视频你懂的在线 | 久久精品免费观看 | 人人要人人澡人人爽人人dvd | 国产小视频在线 | 97久久久免费福利网址 | 美女免费视频一区 | 久久福利| 四虎国产视频 | 欧美日韩一区二区三区不卡 | 久久成人午夜视频 | 丁香花在线观看视频在线 | 亚洲理论片在线观看 | 天天操天天谢 | 久久av免费电影 | 国产日韩欧美综合在线 | 日韩电影一区二区在线 | a在线免费| av在线之家电影网站 | 免费又黄又爽视频 | 在线看黄色av | 日韩黄色软件 | 91精品视频在线免费观看 | 99久久精品国产毛片 | 国产日韩欧美在线免费观看 | 日韩成人免费观看 | 中文字幕一区二区三区在线播放 | 国产91九色视频 | 91亚洲精品久久久久图片蜜桃 | 天天射天天操天天色 | 天天曰天天射 | 九九视频网站 | 成人av免费在线 | 九九热视频在线 | 免费看污在线观看 | 国产精品在线看 | 在线观看不卡视频 | 亚洲精品国产精品国自产 | 免费特级黄毛片 | 色综合久久久久 | 日韩专区在线播放 | 黄p网站在线观看 | 特片网久久 | 99精品在线免费在线观看 | 99精品在线播放 | 色婷婷 亚洲 | 日韩亚洲在线视频 | 亚洲精品视频在线观看免费 | 国产在线精品一区二区不卡了 | 一级a性色生活片久久毛片波多野 | 日韩中文字幕一区 | 日韩av片无码一区二区不卡电影 | 911香蕉视频 | 九九免费在线观看视频 | 91精品国产一区二区三区 | 超碰免费成人 | 在线亚洲免费视频 | 日韩免费在线看 | 国产精品久久久久久久久久久久午夜 | 97日日碰人人模人人澡分享吧 | 九九九国产 | 九九热精品国产 | 亚洲成人av片 | 久久av一区二区三区亚洲 | 97精品国产一二三产区 | 亚洲精品免费观看视频 | 日韩极品视频在线观看 | 久久午夜精品视频 | 国产手机在线精品 | 国产亚洲婷婷 | 青青五月天| 国产精品久久久久久久久久了 | 免费看国产黄色 | 国产精品久久久久久久午夜片 | 久久久久亚洲精品中文字幕 | 国产在线一区观看 | 亚洲精品乱码久久久久 | 尤物97国产精品久久精品国产 | 国产性xxxx | 六月婷婷久香在线视频 | 2019中文最近的2019中文在线 | 午夜av片| 天天操夜夜操国产精品 | 久草国产在线观看 | 波多野结衣电影久久 | 91豆花在线观看 | 人人插人人澡 | 国产乱老熟视频网88av | 国产一级视屏 | 青青河边草免费直播 | www.久久久久 | 在线91观看| 欧美日韩中文视频 | 日韩精品影视 | 在线视频电影 | 欧美大片在线看免费观看 | 最新av电影网址 | 九九九九九精品 | 伊人影院av | 91精品在线播放 | 九九九九九九精品 | 欧美最猛性xxxxx亚洲精品 | 在线激情影院一区 | 狠狠色婷婷丁香六月 | 婷婷网五月天 | 天天干夜夜夜操天 | 国产91在线观看 | 99婷婷狠狠成为人免费视频 | 日韩在线中文字幕 | av在线不卡观看 | 亚洲动漫在线观看 | 国产精品嫩草55av | 在线观看国产福利片 | 日韩欧美精品在线观看视频 | 欧美成人xxxxx | 欧美久久久久久 | 六月婷婷网 | 五月婷婷狠狠 | 日日爽日日操 | 久久精品79国产精品 | 国产精品久久久久久久毛片 | 人人网av | 久草网在线 | 国产字幕在线观看 | 免费日韩视 | 成年人电影免费在线观看 | 五月天免费网站 | av天天色 | 在线a人v观看视频 | 超级碰视频| 香蕉久久久久久av成人 | 日本少妇高清做爰视频 | 国产在线va | 91成年视频| 亚洲一区动漫 | 日韩激情第一页 | 伊人狠狠 | 丁香视频五月 | 激情图片区 | 亚洲欧美国内爽妇网 | 久二影院| 精品99免费 | 一级a性色生活片久久毛片波多野 | 久久国产精品99久久久久久丝袜 | 91在线中文 | 欧美日韩在线观看不卡 | 日日夜日日干 | 日日日操 | 麻豆视频免费入口 | 日日夜夜91 | 久久欧洲视频 | 久久伦理电影 | 69久久久| 亚洲欧美精品一区二区 | 中文免费在线观看 | 国产黄大片 | 亚洲激情电影在线 | 日韩视频区 | 欧洲精品一区二区 | 青青河边草免费直播 | 成人日批视频 | 国产精品久久久久久久久久白浆 | 日韩一区二区在线免费观看 | www.夜夜爱 | 综合五月婷婷 | 国产高清日韩欧美 | 四虎国产精品成人免费影视 | 成年人视频免费在线 | 国产亚洲午夜高清国产拍精品 | 日韩av图片| 亚洲丁香久久久 | 国产不卡av在线播放 | 成人一级片在线观看 | 亚洲激情在线播放 | 成年一级片 | av在线免费不卡 | 国产免费一区二区三区最新 | 日韩精品在线视频 | 亚洲高清在线视频 | 久久久久久高潮国产精品视 | 免费看国产一级片 | 成人a视频片观看免费 | 香蕉视频91 | 国产亚洲精品久久久久久大师 | 黄色一级性片 | 蜜臀av夜夜澡人人爽人人桃色 | 亚洲情影院| 在线观看av国产 | 久操中文字幕在线观看 | 成 人 黄 色 片 在线播放 | 亚洲视频久久久 | 国产欧美日韩精品一区二区免费 | 日日夜夜网 | 中文字幕中文 | 免费观看久久久 | 久久99精品久久久久蜜臀 | 亚洲激情国产精品 | 天堂网在线视频 | 超碰在线资源 | 永久免费毛片在线观看 | 不卡的一区二区三区 | 久久综合九色综合网站 | 人人干人人模 | 国产免费精彩视频 | 日韩成人xxxx| 午夜视频在线观看网站 | 色噜噜色噜噜 | 日韩中文字幕免费视频 | av九九| 亚洲国产欧洲综合997久久, | 免费精品在线视频 | 国产99一区视频免费 | 国产一级免费av | 日韩不卡高清 | 玖玖在线免费视频 | 毛片激情永久免费 | 久在线观看| 亚洲va男人天堂 | 日韩视频免费 | www黄色av| 97超碰人人模人人人爽人人爱 | 欧美另类高清 | 深爱激情五月网 | 久久久久看片 | 国产精品毛片一区二区在线 | 亚洲日日射 | 国产一级高清视频 | 国产美女精品在线 | 狠狠操狠狠操 | 午夜91在线| 国产99久久久国产精品免费二区 | 伊人久久一区 | 国产美女久久久 | 精品久久久久久国产 | 成人在线视频免费观看 | 在线观看免费视频你懂的 | 国产视频综合在线 | 中文字幕在线观看三区 | 亚洲五月婷婷 | 91精品视频免费观看 | 免费日韩一级片 | 亚洲精品免费在线观看视频 | 午夜精品中文字幕 | 六月丁香激情综合色啪小说 | 国产午夜精品av一区二区 | 久久五月网 | 日韩免费高清在线观看 | 久久久久久网站 | 久操操| 99久久精品日本一区二区免费 | 欧美性脚交 | 中文字幕资源站 | 色夜影院 | 五月综合激情婷婷 | 婷婷丁香六月 | 国产一级片网站 | 四季av综合网站 | 国产亚洲久一区二区 | 99视频导航 | 91麻豆精品国产自产在线 | 精品久久久久久综合日本 | 美女久久一区 | 久久综合五月天 | 天天激情站 | 国产视频在线播放 | 黄色大片视频网站 | 午夜久久福利 | 精品影院一区二区久久久 | 久久久影院官网 | av888av.com| 色视频在线观看 | www黄色com| 一区二区三区四区精品 | 黄污在线观看 | 啪啪免费试看 | 天堂av免费在线 | 成人久久精品视频 | 91av视频网站| 国产成人亚洲在线电影 | 性色av香蕉一区二区 | 国产成人精品在线 | 免费观看一级 | 99re国产视频 | 国产伦理久久 | 色婷婷综合久久久久 | aa级黄色大片 | 九九影视理伦片 | 亚洲免费精品一区二区 | 久久视屏网 | 日韩精品亚洲专区在线观看 | 91夫妻视频| 精品专区一区二区 | 色天天久久 | 欧美日韩精品在线观看视频 | 在线免费观看黄色大片 | 在线成人小视频 | 综合在线亚洲 | 日韩欧美在线高清 | 亚洲色视频 | 久久久蜜桃 | 69视频网站| 九九视频一区 | 欧美日韩一区二区三区在线免费观看 | 人人爽人人爽av | 手机版av在线 | 亚洲欧洲国产精品 | 激情五月婷婷激情 | 天天草天天| 国产91精品一区二区麻豆网站 | 中文字幕成人在线 | 九九免费在线视频 | 人人看人人爱 | 九九在线免费视频 | 精品影院一区二区久久久 | 国产午夜精品理论片在线 | 国产精品国产亚洲精品看不卡15 | 免费a网站 | 久久国色夜色精品国产 | 欧美日韩一区二区三区在线免费观看 | 欧美精品久久久久久久免费 | 91九色在线观看视频 | 色综合a | .国产精品成人自产拍在线观看6 | av夜夜操| 一级精品视频在线观看宜春院 | 激情在线免费视频 | 国产成人一区二区三区电影 | 午夜12点| 精品在线视频播放 | 天天操天天摸天天干 | 国产群p视频 | 久久久久久久久久久电影 | 国产成人精品午夜在线播放 | 国产精品美女免费视频 | 亚洲精品午夜aaa久久久 | 日日夜日日干 | 久久久久久久久亚洲精品 | 欧美激情第八页 | 国产一级精品视频 | 97在线视| 欧美在线视频一区二区 | 五月婷婷天堂 | 福利视频网站 | 色中射 | 狠狠色网 | 黄色小说视频在线 | 高潮久久久久久久久 | 日韩大片在线看 | 在线视频欧美亚洲 | 色偷偷中文字幕 | 亚洲.www | 久久精国产| 久久这里只有精品首页 | 国产精美视频 | 日韩av中文在线 | 2020天天干夜夜爽 | 久久久久综合精品福利啪啪 | 国产成人一区二区三区久久精品 | 婷婷丁香五 | 久久精品视频一 | 人人澡人人模 | 9幺看片 | 久草视频免费看 | a级成人毛片| 成人午夜免费剧场 | 91在线免费播放视频 | 国产在线中文 | 国产成人精品一区二区三区在线观看 | 天天操天天操天天爽 | 成人h电影在线观看 | www.国产精品 | 视频一区在线免费观看 | 五月开心六月婷婷 | 丁香五月亚洲综合在线 | 91亚洲精品乱码久久久久久蜜桃 | 国产丝袜一区二区三区 | 国产在线理论片 | 国产欧美精品一区二区三区 | 999久久久久久久久久久 | 国产91影院| 视频一区二区国产 | 国产精品九九久久久久久久 | 亚洲视频一级 | 国内久久久久久 | 丁香狠狠 | 亚洲另类视频在线观看 | 国产精品欧美日韩在线观看 | 国产午夜三级一区二区三桃花影视 | 狠狠色丁香婷婷综合橹88 | 欧美福利精品 | 久草视频免费在线观看 | 美女网站黄免费 | 黄色天堂在线观看 | 麻豆av电影 | 中文字幕综合在线 | 久久综合久色欧美综合狠狠 | 国产中文字幕精品 | 国产一二区视频 | 天天操偷偷干 | 日本久久成人 | 色婷五月天 | 国产精品毛片一区二区 | 四虎最新域名 | 国内丰满少妇猛烈精品播 | 国产专区精品视频 | 成人a v视频 | 婷婷在线观看视频 | jizz18欧美18 | 91精彩视频在线观看 | 男女啪啪网站 | 久久国产精品视频免费看 | 91福利视频久久久久 | 中文字幕亚洲五码 | 久久久国产精品免费 | 久久久久亚洲a | 婷婷六月天综合 | 伊人色**天天综合婷婷 | 国产手机在线精品 | 在线免费观看黄色 | 久久精品在线视频 | 午夜精品久久久久久久99 | 99精品免费久久久久久久久日本 | 成人免费一级片 | 久久综合射| 久久九九视频 | 色欧美视频 | 欧美一区二区三区在线观看 | www免费在线观看 | 天天操人人要 | 久久精品国产免费看久久精品 | 亚洲在线色 | 国产色综合天天综合网 | 中文字幕资源网 国产 | 亚洲激情六月 | 特级西西444www大胆高清无视频 | 黄色在线看网站 | 制服丝袜在线91 | 午夜影院在线观看18 | 欧美日高清视频 | 91午夜精品 | 丁香视频全集免费观看 | 香蕉在线观看 | 91少妇精拍在线播放 | 69视频国产 | 夜添久久精品亚洲国产精品 | 黄色的视频| 国产精品99久久久久 | 激情 婷婷| 久久成人综合视频 | 亚洲成av人片 | 中文资源在线官网 | 97在线影视 | 狠狠色噜噜狠狠狠狠2022 | 久日精品 | 日本精品久久久久 | 亚洲va在线va天堂va偷拍 | 在线国产片 | 探花国产在线 | 天天操天天干天天插 | 国产精品视频大全 | 成人教育av | 久久黄色免费视频 | 日日夜夜天天 | 国产精品午夜在线观看 | 日韩av一区在线观看 | 色视频在线免费 | 成人丁香花| 中文字幕在线观看一区 | 永久免费观看视频 | 不卡av在线免费观看 | 欧美天天综合网 | 国产一区二区免费 | 国产亚洲成人网 | 久久久黄视频 | 国产精品久久精品国产 | 自拍超碰在线 | 91欧美视频网站 | 最近2019中文免费高清视频观看www99 | 久草影视在线观看 | 亚洲特级片 | 欧美成人精品欧美一级乱 | 日韩高清毛片 | 成人av一区二区三区 | 日韩视频在线不卡 | 国产色婷婷精品综合在线手机播放 | 久久精品伊人 | 成年美女黄网站色大片免费看 | 亚洲伊人网在线观看 | 91漂亮少妇露脸在线播放 | 91在线视频免费91 | 欧美一区二区精美视频 | 狠狠色丁香久久婷婷综 | 热久在线 | 在线视频亚洲 | 亚洲理论在线 | 免费麻豆 | 97天天干| 久久婷婷丁香 | 国产香蕉视频在线观看 | 亚洲乱码国产乱码精品天美传媒 | 午夜精品电影一区二区在线 | 中文在线www| 日韩午夜大片 | www.天天色.com | 日日碰狠狠躁久久躁综合网 | 中文字幕日韩精品有码视频 | 成人app在线免费观看 | 欧美在线观看小视频 | 日韩精品视频在线观看免费 | 久久亚洲私人国产精品va | 亚洲精品视频在线播放 | 亚洲女在线 | 国产男女无遮挡猛进猛出在线观看 | 国产日韩精品一区二区三区 | 日本黄网站 | 日韩视频一二三区 | 综合网伊人| 成人黄色电影在线播放 | 91精品国产成人www | 国内精品久久久精品电影院 | av在线播放不卡 | 色五月激情五月 | 成人a免费视频 | 国产麻豆视频免费观看 | 国产精品久久久久久婷婷天堂 | 日韩午夜网站 | 在线观看黄色免费视频 | 亚洲美女精品区人人人人 | 黄色大片网 | 免费av一级电影 | 超碰人人91 | 国产亚洲精品久久久久久电影 | 久久久亚洲网站 | 久久久九九 | 草久久久久 | 久久久综合精品 | 91成版人在线观看入口 | 婷婷国产在线 | 久久成人18免费网站 | 99精品亚洲 | 日韩电影中文字幕在线 | 青青河边草观看完整版高清 | 五月黄色 | 亚洲精品国 | 久草在线免费色站 | 91资源在线播放 | 天天天天干 | 激情五月婷婷综合网 | 欧美日韩国产一区二区三区 | 天天干天天做 | 黄色一级大片在线观看 | 在线看的毛片 | 正在播放久久 | 国产免费高清视频 | 国产精品久久久久久999 | 国产精品美女免费看 | 99久久精品免费看国产一区二区三区 | 国产麻豆成人传媒免费观看 | 久久精品电影院 | 国产精品视频免费观看 | 日日草视频 | 一区二区视频网站 | 日韩欧美第二页 | 13日本xxxxxⅹxxx20 | 精品一区二区视频 | 91久久精品日日躁夜夜躁国产 | 97国产小视频 | 欧美日韩国产一区二区在线观看 | 成人av电影免费在线观看 | 伊人天堂久久 | 日韩福利在线观看 | 黄色日视频 | 国产高清久久 | 日韩激情第一页 | 亚洲成人网在线 | 欧美日韩在线视频一区二区 | 特级a老妇做爰全过程 | 夜夜操综合网 | 国产精品6 | 91在线色| 日韩有码在线观看视频 | 久久精精品视频 | 国产成人一区二区精品非洲 | 免费在线激情视频 | www黄在线| 偷拍精偷拍精品欧洲亚洲网站 | 欧美亚洲一区二区在线 | 亚洲九九九在线观看 | 天天爽天天射 | 亚洲夜夜网 | 国产精品美女久久久久久久久 | 在线综合色 | 日韩成人免费在线观看 | 久久成人麻豆午夜电影 | 国产正在播放 | 亚洲人成免费网站 | 天天操夜夜曰 | 中文字幕av免费在线观看 | 久久99欧美 | 亚洲精品字幕在线 | 91中文字幕在线观看 | 日韩免费不卡av | 91高清免费在线观看 | 国产视频一区二区在线 | www.玖玖玖 | 国产精品99久久久精品 | 人人射 | 在线看一级片 | 麻豆视频在线免费看 | 亚洲午夜久久久久 | 亚洲区另类春色综合小说 | 日韩草比 | 成人免费观看完整版电影 | 岛国片在线 | 色婷婷成人 | 欧美日韩国产区 | 99久久精品免费看国产免费软件 | 婷婷成人在线 | 欧美日韩国产综合一区二区 | 亚洲无吗av | 开心激情五月网 | 日韩精品在线看 | 92av视频| 亚洲欧美国产精品 | 亚洲狠狠干 | 黄色三级视频片 | 国产精品久久久免费看 | 亚洲欧洲成人 | 日韩精品一区二区三区高清免费 | 免费91在线 | 中文字幕日韩有码 | 国产精品黄网站在线观看 | 91亚洲综合 | 国内久久视频 | 久久精品2 | 日韩精品视频免费在线观看 | av官网 | 国产精品一区二区三区电影 | 怡红院av久久久久久久 | 欧美日韩午夜 | 国产精品毛片久久久久久 | 国产无吗一区二区三区在线欢 | 国产精品午夜久久 | 欧美老女人xx | 亚洲91中文字幕无线码三区 | 天堂黄色片 | 日韩午夜视频在线观看 | 波多野结衣电影一区二区三区 | 免费看一及片 | 国产欧美最新羞羞视频在线观看 | 精品在线观看一区二区 | 在线观看色网站 | 久久精品99视频 | 久久综合五月 | 欧美久久久久久久久久 | 精品免费视频123区 午夜久久成人 | 婷婷丁香在线 | 91超碰免费在线 | 日韩精品视频免费在线观看 | www.天天射| 久久电影国产免费久久电影 | 免费看黄色小说的网站 | 久久中国精品 | 国产高清区 | 国产精品99久久久久久久久 | 色网站在线免费 | 免费观看www小视频的软件 | 狠狠狠狠狠狠 | 日韩精品一区二区三区免费视频观看 | 天天操夜夜叫 | 色综合人人 | 亚洲欧洲精品一区二区 | 最新av免费在线观看 | 国产精成人品免费观看 | 国产中的精品av小宝探花 | 日韩成人看片 | 久久综合99 | 人人看人人爱 | 国产日本高清 | 狠狠色丁香久久婷婷综合五月 | a级国产乱理论片在线观看 伊人宗合网 | 黄色网在线播放 | 天天躁天天操 | 成人免费网视频 | 亚洲精品自拍视频在线观看 | 免费大片av| 国产精品毛片一区视频播不卡 | 亚洲欧洲精品一区 | av综合网址 | 国产亚洲精品久久久久久网站 | 丁香婷婷亚洲 | 欧美日韩99 | 成人黄色av网站 | 日产中文字幕 | 精品国模一区二区 | 国产精品18久久久久白浆 | 天天操夜夜摸 | 色99之美女主播在线视频 | 久久99网站 | 在线观看麻豆av | 91精品国产99久久久久久久 | 日韩在线观看中文 | 免费看v片| 伊人视频| 深夜成人av | 亚洲精品合集 | 日韩精品久久久久久 | 天天综合五月天 | 伊人首页| 日韩精品中文字幕在线不卡尤物 | 一区中文字幕在线观看 | 国产精品美女久久久久久免费 | 国产精品一区二区三区久久 | 免费观看www小视频的软件 | 国内一级片在线观看 | 黄色软件网站在线观看 | 成人中文字幕+乱码+中文字幕 | 亚洲一二三在线 | 欧美日韩啪啪 | 久久久久久国产精品 | 亚洲精品九九 | 中文字幕成人av | 国产色婷婷精品综合在线手机播放 | 九色视频网站 | 天堂在线视频中文网 | 欧美日韩高清在线观看 | 中字幕视频在线永久在线观看免费 | 中文字幕视频免费观看 | 精品爱爱| 色婷婷av一区二 | 综合成人在线 | 免费看成人a | 美女久久 | 亚洲一区二区精品在线 | 草久久av| 日韩视频一区二区在线 | 国产伦精品一区二区三区… | 成人黄色影片在线 | 欧美高清视频不卡网 | 成人午夜在线电影 | 国产一区不卡在线 | 欧美精品国产综合久久 | 欧美aa级| 久久99视频| 91超级碰 | 欧美99热 | 日韩视频一区二区在线观看 | 日韩成人中文字幕 | 国产手机视频在线 | 在线观看视频99 | 一级黄色毛片 | 成人网色 | 日韩中文字幕在线 | 97超碰国产在线 | 韩国在线一区二区 | 久久网站最新地址 | 亚洲欧美在线视频免费 | 国产成人无码AⅤ片在线观 日韩av不卡在线 | 日韩中文字幕在线不卡 | 亚洲最新在线视频 | 国产精品午夜在线观看 | 日韩视频一区二区在线 | 日韩色爱 | 亚洲黄色成人av | av午夜电影 | 四虎影视国产精品免费久久 | 狠狠干夜夜操天天爽 | 美女网站在线免费观看 | 久久久久久久久久久国产精品 | 日韩精品一区二区三区高清免费 | 一区二区三区 亚洲 | 国产精品九九热 | 婷婷五月色综合 | 国产精品久久久久久久久婷婷 | 久久视频在线观看免费 | 在线播放视频一区 | 91丨九色丨国产丨porny精品 | 六月色婷 | 黄网站色欧美视频 | 草久在线 | 成人h视频在线播放 | 美女免费视频观看网站 | 国产99久久久精品 | 国产黄色在线观看 | 免费看国产曰批40分钟 | 久热免费在线 | 日韩视频免费在线观看 | 伊人中文在线 | 91资源在线观看 | 激情视频一区二区三区 | 日韩高清免费电影 | 日韩精品视频免费看 | 一区二区三区四区五区在线 | 天天爽夜夜爽人人爽一区二区 | 亚洲精品免费在线播放 | 91在线区 | 国产亚洲视频中文字幕视频 | 国产资源在线视频 | 精品亚洲视频在线观看 | 国产一区二区在线免费 | 亚洲欧洲中文日韩久久av乱码 | 天天干夜夜夜 | 国产精品2区 | 丁香婷婷久久久综合精品国产 | 国语麻豆 | 久久人人插 | 一区中文字幕在线观看 | 在线观看香蕉视频 | 久久精品一区二区 | 亚洲激情在线视频 | 久久成人麻豆午夜电影 | 国产资源av| 国产精品免费av | 蜜臀av性久久久久蜜臀aⅴ流畅 | 亚洲特级片 | 日韩特级毛片 | 亚洲精品在线电影 | 成人av片免费看 | 超碰久热| 成人久久久久久久久久 | 在线欧美a | 亚洲黄色一级大片 | 高清久久久 | 国产精品国产自产拍高清av | 日韩成人免费在线观看 | 亚洲香蕉在线观看 | 美女视频一区二区 | 日韩高清精品免费观看 | 99热精品国产一区二区在线观看 | 国产伦理久久精品久久久久_ | 最新一区二区三区 | 色婷婷免费| 日韩欧美国产精品 | 中文区中文字幕免费看 | 日韩高清成人在线 | 人人澡人人爽欧一区 | 亚洲精品电影在线 | 肉色欧美久久久久久久免费看 |