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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 人文社科 > 生活经验 >内容正文

生活经验

《深入理解Android:Wi-Fi,NFC和GPS》章节连载[节选]--第二章 深入理解Netd

發布時間:2023/11/27 生活经验 52 豆豆
生活随笔 收集整理的這篇文章主要介紹了 《深入理解Android:Wi-Fi,NFC和GPS》章节连载[节选]--第二章 深入理解Netd 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

原文: http://blog.csdn.net/innost/article/details/20400389

本文使用的源碼版本為Android5.1

本章主要內容
介紹Netd;
介紹MDNS和Apple Bonjour技術;
介紹iptables、tc和ip等Linux系統中常用的網絡管理工具;
介紹Netd中的各個命令對象和相關的背景知識;
介紹NetworkManagmentService。
2.1 概述
Netd是Android系統中專門負責網絡管理和控制的后臺daemon程序,其功能主要分三大塊:
設置防火墻(Firewall)、網絡地址轉換(NAT)、帶寬控制、無線網卡軟接入點(Soft Access Point)控制,網絡設備綁定(Tether)等。
Android系統中DNS信息的緩存和管理。
網絡服務搜索(Net Service Discovery,簡稱NSD)功能,包括服務注冊(Service Registration)、服務搜索(Service Browse)和服務名解析(Service Resolve)等。
Netd的工作流程和Vold類似[1],其工作可分成兩部分:
Netd接收并處理來自Framework層中NetworkManagementService或NsdService的命令。這些命令最終由Netd中對應的Command對象去處理。
Net接收并解析來自Kernel的UEvent消息,然后再轉發給Framework層中對應Service去處理。
由上述內容可知,Netd位于Framework層和Kernel層之間,它是Android系統中網絡相關消息和命令轉發及處理的中樞模塊。
Netd的代碼量不大,難度較低,但其所涉及的相關背景知識卻比較多。本章對Netd的分析將從以下幾個方面入手:
首先介紹Netd的大體工作流程以及DNS、MDns相關的背景知識。關于Netd的工作流程分析,讀者也可參考①中的內容。
然后本章將集中介紹Netd中涉及到的Android系統中網絡管理和控制的相關工具。它們是iptables、tc和ip。
最后將介紹Netd中CommandListener的命令處理。這些命令的正常工作依賴于上面介紹的iptables等工具。
最后,我們將介紹Java Framework中的NetworkManagementService服務。
提示:NsdService比較簡單,感興趣的讀者不妨閱讀作者的一篇博文”Android Says Bonjour”中的第2.2“NsdService介紹”一節。地址位于 http://blog.csdn.net/innost/article/details/8629139。
2.2 Netd工作流程分析
Netd進程由init進程根據init.rc的對應配置項[1]而啟動,其配置項為:

service netd /system/bin/netdclass mainsocket netd stream 0660 root systemsocket dnsproxyd stream 0660 root inetsocket mdns stream 0660 root system
Netd啟動時將創建三個TCP監聽socket,其名稱分別為"netd","dnsproxyd"和"mdns"。
根據本章后續分析,讀者將會看到:
Framework層中的NetworkManagementService和NsdService將分別和"netd"及"mdns"監聽socket建立鏈接并交互。
每一個調用和域名解析相關的socket API(如getaddrinfo或gethostbyname等)的進程都會借由"dnsproxyd"監聽socket與netd建立鏈接。
下面開始分析Netd進程。
2.2.1 main函數分析
Netd進程的入口函數是其main函數,代碼如下所示:

system/netd/server/main.cpp

int main() {CommandListener *cl;NetlinkManager *nm;DnsProxyListener *dpl;MDnsSdListener *mdnsl;FwmarkServer* fwmarkServer;ALOGI("Netd 1.0 starting");remove_pid_file();blockSigpipe();//①創建NetlinkManagerif (!(nm = NetlinkManager::Instance())) {ALOGE("Unable to create NetlinkManager");exit(1);};//②創建CommandListener,它將創建名為"netd"的監聽socketcl = new CommandListener();nm->setBroadcaster((SocketListener *) cl);if (nm->start()) {ALOGE("Unable to start NetlinkManager (%s)", strerror(errno));exit(1);}// Set local DNS mode, to prevent bionic from proxying// back to this service, recursively.//注意下面這行代碼,它為本Netd設置環境變量ANDROID_DNS_MODE為"local",其作用將在2.2.4節介紹setenv("ANDROID_DNS_MODE", "local", 1);//③創建DnsProxyListener,它將創建名為"dnsproxyd"的監聽socketdpl = new DnsProxyListener(CommandListener::sNetCtrl);if (dpl->startListener()) {ALOGE("Unable to start DnsProxyListener (%s)", strerror(errno));exit(1);}//④創建MDnsSdListener并啟動監聽,它將創建名為"mdns"的監聽socketmdnsl = new MDnsSdListener();if (mdnsl->startListener()) {ALOGE("Unable to start MDnsSdListener (%s)", strerror(errno));exit(1);}fwmarkServer = new FwmarkServer(CommandListener::sNetCtrl);if (fwmarkServer->startListener()) {ALOGE("Unable to start FwmarkServer (%s)", strerror(errno));exit(1);}/** Now that we're up, we can respond to commands*/if (cl->startListener()) {ALOGE("Unable to start CommandListener (%s)", strerror(errno));exit(1);}bool wrote_pid = write_pid_file();while(1) {sleep(30); // 30 secif (!wrote_pid) {wrote_pid = write_pid_file();}}ALOGI("Netd exiting");remove_pid_file();exit(0);
}
NM的start函數主要是向Kernel注冊了三個用于接收UEvent事件的socket,這三個UEvent[1][2]分別對應于:

NETLINK_KOBJECT_UEVENT:代表kobject事件,由于這些事件包含的信息由ASCII字符串表達,故上述代碼中使用了NETLINK_FOMRAT_ASCII。它表示將采用字符串解析的方法去解析接收到的UEvent消息。kobject一般用來通知內核中某個模塊的加載或卸載。對NM來說,其關注的是/sys/class/net下相應模塊的加載或卸載消息。
NETLINK_ROUTE:代表kernel中routing或link改變時對應的消息。NETLINK_ROUTE包含很多子項,上述代碼中使用了RTMGRP_LINK項。二者結合起來使用,表示NM希望收到網絡鏈路斷開或接通時對應的UEvent消息(筆者在Ubuntu PC機上測試過,當網卡上拔掉或插入網線時,會觸發這些UEvent消息的發送)。由于對應UEvent消息內部封裝了nlmsghdr等相關結構體,故上述代碼使用了NETLINK_FORMAT_BINARY來指示解析UEvent消息時將使用二進制的解析方法。
NETLINK_NFLOG:和2.3.6節介紹的帶寬控制有關。Netd中的帶寬控制可以設置一個預警值,當網絡數據超過一定字節數就會觸發kernel發送一個警告。該功能屬于iptables的擴展項,但由于iptables的文檔更新速度較慢(這也是很多開源項目的一大弊端),筆者一直未能找到相關的正式說明。值得指出的是,上述代碼中有關NETLINK_NFLOG相關socket的設置并非所有kernel版本都支持。同時,NFLOG_QUOTA_GROUP的值是直接定義在NetlinkManager.cpp中的,而非和其他類似系統定義一樣定義在系統頭文件中。這也表明NFLOG_QUOTA_GROUP的功能比較新。
提示:讀者可通過在Linux終端中執行man PF_LINK得到有關NETLINK的詳細說明。
上述start函數將調用setupSocket創建用于接收UEvent消息的socket以及一個解析對象NetlinkHandler。setupSocket代碼本身比較簡單,此處就不擬展開分析。
下面來看NM及其家族成員,它們之間的關系如圖2-2所示。

圖2-2 NetlinkManager家族成員的類圖
由圖2-2可知:
NetlinkHandler和CommandListener均間接從SocketListener派生。其中,NetlinkHandler收到的socket消息將通過onEvent回調處理。
結合前文所述,NetlinkManager分別注冊了三個用于接收UEvent的socket,其對應的NetlinkHandler分別是mUeventHandler、mRouteHandler和mQuotaHandler。
NetlinkHandler接收到的UEvent消息會轉換成一個NetlinkEvent對象。NetlinkEvent對象封裝了對UEvent消息的解析方法。對于NETLINK_FOMRAT_ASCII類型,其parseAsciiNetlinkMessage函數會被調用,而對于NETLINK_FORMAT_BINARY類型,其parseBinaryNetlinkMessage函數會被調用。
NM處理流程的輸入為一個解析后的NetlinkEvent對象。NM完成相應工作后,其處理結果將經由mBroadcaster對象傳遞給Framework層的接收者,也就是NetworkManagementService。
CommandListener從FrameworkListener派生,而FrameworkListener內部有一個mCommands數組,它用來存儲注冊到FrameworkListener中的命令處理對象。
下面來簡單了解下NetlinkHandler的onEvent函數,由于其內部已針對不同屬性的NetlinkEvent進行了分類處理,故瀏覽這段代碼能加深對前文所述不同UEvent消息的作用的理解。
[-->NetlinkHandler.cpp::onEvent]
void NetlinkHandler::onEvent(NetlinkEvent *evt) {
const char *subsys = evt->getSubsystem();
......
//處理對應NETLINK_KOBJECT_UEVENT和NETLINK_ROUTE的信息
if (!strcmp(subsys, "net")) {
int action = evt->getAction();
const char *iface = evt->findParam("INTERFACE");//查找消息中攜帶的網絡設備名
if (action == evt->NlActionAdd) {
notifyInterfaceAdded(iface);//添加NIC(Network Interface Card)的消息
} else if (action == evt->NlActionRemove) {
notifyInterfaceRemoved(iface);//NIC被移除的消息
} else if (action == evt->NlActionChange) {
evt->dump();
notifyInterfaceChanged("nana", true);//NIC變化消息
} else if (action == evt->NlActionLinkUp) {//下面兩個消息來自NETLINK_ROUTE
notifyInterfaceLinkChanged(iface, true);//鏈路啟用(類似插網線)
} else if (action == evt->NlActionLinkDown) {
notifyInterfaceLinkChanged(iface, false);//鏈路斷開(類似拔網線)
}
} else if (!strcmp(subsys, "qlog")) {//對應NETLINK_NFLOG
const char *alertName = evt->findParam("ALERT_NAME");
const char *iface = evt->findParam("INTERFACE");
notifyQuotaLimitReached(alertName, iface);//當數據量超過預警值,則會收到該通知
} else if (!strcmp(subsys, "xt_idletimer")) {
//這個和后文的idletimer有關,用于跟蹤某個NIC的工作狀態,即是“idle”還是“active”
//檢測時間按秒計算
int action = evt->getAction();
const char *label = evt->findParam("LABEL");
const char *state = evt->findParam("STATE");
if (label == NULL) {
label = evt->findParam("INTERFACE");
}
if (state)
notifyInterfaceClassActivity(label, !strcmp("active", state));
}
......
}
由上邊代碼可知:
NETLINK_KOBJECT_UEVENT和NETLINK_ROUTE主要反映網絡設備的事件和狀態,包括NIC的添加、刪除和修改,以及鏈路的連接狀態等。
NETLINK_NFLOG用于反映設置的log是否超過配額。
另外,上邊代碼中還處理了“xt_idletimer”的uevent消息,它和后文要介紹的IdleTimerCmd有關,主要用來監視網絡設備的收發工作狀態。當對應設備工作或空閑時間超過設置的監控時間后,Kernel將會發送攜帶其狀態("idle"或"active")的UEvent消息。
圖2-3所示為NetlinkHandler的工作流程。

圖2-3 NM工作流程圖
由圖2-3可知:
NM創建NetlinkHandler后,工作便轉交給NetlinkHandler來完成,而每個NetlinkHandler對象均會單獨創建一個線程用于接收Socket消息。
當Kernel發送UEvent消息后,NetlinkHandler便從select調用中返回,然后調用其onDataAvailable函數,該函數內部會創建一個NetlinkEvent對象。
NetlinkEvent對象根據socket創建時指定的解析類型去解析來自Kernel的UEvent消息。
最終NetlinkHandler的onEvent將被調用,不同的UEvent消息將在此函數中進行分類處理。
NetlinkHandler最終將處理結果經由NM內部變量mBroadcaster轉發給NetworkManagementService。
提醒:請讀者結合上文所述流程自行研讀相關代碼。

[1]關于init工作原理以及init.rc的分析方法,讀者可參考《深入理解Android:卷1》第3章關于init進程的分析。

[1]讀者可參考《深入理解Android:卷1》第9章關于Vold的分析。
2.2.3 CommandListener分析
Netd中第二個重要成員是CommandListener(以后簡稱CL),其主要作用是接收來自Framework層NetworkManageService的命令。從角色來看,CL僅是一個Listener。它在收到命令后,只是將它們轉交給對應的命令處理對象去處理。CL內部定義了許多命令,而這些命令都有較深的背景知識。本節擬以分析CL的工作流程為主,而相關的命令處理則放到后文再集中分析。
CL中的圖2-4所示為CL中的Command對象及對應的Controller對象。

圖2-4 CL中的命令及控制類
由圖2-4可知:
CL定義了11個和網絡相關的Command類。這些類均從NetdCommand派生(注意,為保持繪圖簡潔,這11個Command的派生關系由1個派生箭頭表達)。
CL還定義了10個控制類,這些控制類將和命令類共同完成相應的命令處理工作。
結合前面圖2-2中對NM家族成員的介紹,CL創建時,需要注冊自己支持的命令類。這部分代碼在其構造函數中實現,代碼如下所示:
[-->CommandListener::CommandListener構造函數]
CommandListener::CommandListener() :
FrameworkListener("netd", true) {
registerCmd(new InterfaceCmd());//注冊11個命令類對象
registerCmd(new IpFwdCmd());
registerCmd(new TetherCmd());
registerCmd(new NatCmd());
registerCmd(new ListTtysCmd());
registerCmd(new PppdCmd());
registerCmd(new SoftapCmd());
registerCmd(new BandwidthControlCmd());
registerCmd(new IdletimerControlCmd());
registerCmd(new ResolverCmd());
registerCmd(new FirewallCmd());
//創建對應的控制類對象
if (!sSecondaryTableCtrl)
sSecondaryTableCtrl = new SecondaryTableController();
if (!sTetherCtrl)
sTetherCtrl = new TetherController();
if (!sNatCtrl)
sNatCtrl = new NatController(sSecondaryTableCtrl);
if (!sPppCtrl)
sPppCtrl = new PppController();
if (!sSoftapCtrl)
sSoftapCtrl = new SoftapController();
if (!sBandwidthCtrl)
sBandwidthCtrl = new BandwidthController();
if (!sIdletimerCtrl)
sIdletimerCtrl = new IdletimerController();
if (!sResolverCtrl)
sResolverCtrl = new ResolverController();
if (!sFirewallCtrl)
sFirewallCtrl = new FirewallController();
if (!sInterfaceCtrl)
sInterfaceCtrl = new InterfaceController();
//其他重要工作,后文再分析
}
由于CL的間接基類也是SocketListener,所以其工作流程和NetlinkHandler類似。
為了方便讀者理解,圖2-5給出了CL的工作流程圖:

圖2-5 CL的工作流程示意圖
圖2-5中,假設Client端發送的命令名是"nat",當CL收到這個命令后,首先會從其構造函數中注冊的那些命令對象中找到對應該名字(即"nat")的命令對象,其結果就是圖中的NatCmd對象。而該命令最終的處理工作將由此NatCmd對象的runCommand函數完成。
2.2.4 DnsProxyListener分析
DnsProxyListener和Android系統中的DNS管理有關。什么是DNS呢?Android系統中DNS又有什么特點呢?來看下文。
1. Android DNS介紹[3]
DNS是Domain Name System(域名系統)的縮寫。其主要目的是在域名和IP地址之間建立一種映射。簡單點說,DNS的功能類似于電話簿,它可將人名映射到相應的電話號碼。在DNS中,人名就是域名,電話號碼就是IP地址。域名系統的管理由DNS服務器來完成。全球范圍內的DNS服務器共同構成了一個分布式的域名-IP數據庫。
對使用域名來發起網絡操作的網絡程序來說,其域名解析工作主要分兩步:
1)第一步工作就是需要將域名轉換成IP。由于域名和IP的轉換關系存儲在DNS服務器上,所以該網絡程序要向DNS服務器發起請求,以獲取域名對應的IP地址。
2)DNS服務器根據DNS解析規則解析并得到該域名對應的IP地址,然后返回給客戶端。在DNS中,每一個域名和IP的對應關系被稱之為一條記錄。客戶端一般會緩存這條記錄以備后續之用。
提醒:DNS解析規則比較復雜,感興趣的讀者可研究DNS的相關協議。
對軟件開發者來說,常用的域名解析socket API有兩個:
getaddrinfo:它根據指定的host名或service名得到對應的IP地址(該IP地址由結構體addrinfo表達)。
getnameinfo:根據指定的IP地址(由結構體sockaddr表達)得到對應的host或service的名稱。
Android中,這兩個函數均由Bionic C實現。其代碼實現基于NetBSD的解析庫(resolver library),并經過一些修改。這些修改包括:
沒有實現name-server-switch功能。這是為了保持Bionic C庫的輕便性而做的裁剪。
DNS服務器的配置文件由/etc/resolv.conf變成/system/etc/resolv.conf[1]。在Android系統中,/etc目錄實際上為/system/etc目錄的鏈接。resolv.conf存儲的是DNS服務器的IP地址。
系統屬性中保存了一些DNS服務器的地址,它們通過諸如"net.dns1"或"net.dns2"之類的屬性來表達。這些屬性由dhcpd進程或其他系統模塊負責維護。
每個進程還可以設置進程特定的DNS服務器地址。它們通過諸如"net.dns1.<pid>"或"net.dns2.<pid>"的系統屬性來表達。
不同的網絡設備也有對應的DNS服務器地址,例如通過wlan接口發起的網絡操作,其對應的DNS服務器由系統屬性“net.wlan.dns1”表示。
圖2-6所示為三星Galaxy Note2中有關dns信息的示意圖。

圖2-6 net.dns設置示意圖
由圖2-6可知:
系統中有些進程有自己特定的DNS服務器。
不同網絡設備也設置了對應的DNS服務器地址。
2. getaddrinfo函數分析
本節將介紹Android中getaddrinfo的實現,我們將只關注Android對其做的改動。
[-->getaddrinfo.c::getaddrinfo]
int getaddrinfo(const char *hostname, const char *servname,
const struct addrinfo *hints, struct addrinfo **res)
{
......//getaddrinfo的正常處理
//Android平臺的特殊定制
if (android_getaddrinfo_proxy(hostname, servname, hints, res) == 0) {
return 0;
}
......//如果上述函數處理失敗,則繼續getaddrinfo的正常處理
return error
}
由上述代碼可知,Android平臺中的getaddrinfo會調用其定制的android_getaddrinfo_proxy函數完成一些特殊操作,該函數的實現如下所示:
[-->getaddrinfo.c::android_getaddrinfo_proxy]
static int android_getaddrinfo_proxy(const char *hostname, const char *servname,
const struct addrinfo *hints, struct addrinfo **res)
{
.......
//取ANDROID_DNS_MODE環境變量。只有Netd進程設置了它
const char* cache_mode = getenv("ANDROID_DNS_MODE");
......
//由于Netd進程設置了此環境變量,故Netd進程調用getaddrinfo的話,將不會采用這套定制的方法
if (cache_mode != NULL && strcmp(cache_mode, "local") == 0) {
return -1;
}
//獲取本進程對應的DNS地址
snprintf(propname, sizeof(propname), "net.dns1.%d", getpid());
if (__system_property_get(propname, propvalue) > 0) {
return -1;
}

//建立和Netd中DnsProxyListener的連接,將請求轉發給它去執行
sock = socket(AF_UNIX, SOCK_STREAM, 0);
if (sock < 0) {
return -1;
}
......
strlcpy(proxy_addr.sun_path, "/dev/socket/dnsproxyd",
sizeof(proxy_addr.sun_path));
......//發送請求,處理回復等
return -1;
}
由上述代碼可知:
當Netd進程調用getaddrinfo時,由于其設置了ANDROID_DNS_MODE環境變量,所以該函數會繼續原來的流程。
當非Netd進程調用getaddrinfo函數時,首先會開展android_getaddrinfo_proxy中的工作,即判斷該進程是否有定制的DNS服務器,如果沒有的話它將和位于Netd進程中的"dnsproxyd"監聽socket建立連接,然后把請求發給DnsProxyListener去執行。
3. DnsProxyListener命令介紹
下面來介紹DnsProxyListener(以后簡稱DPL),圖2-7所示為其家族成員示意圖:

圖2-7 DPL家族示意圖
由圖2-7可知,DPL僅定義了兩個命令:
GetAddrInfoCmd,和Bionic C庫的getaddrinfo函數對應。
GetHostByAddrCmd,和Bionic C庫的gethostbyaddr函數對應。
這個兩條命令的處理比較簡單,此處就不擬展開詳細的代碼。
為方便讀者理解,我們將給出調用序列圖,如圖2-8所示。

圖2-8 GetAddrInfoCmd處理流程示意圖
由圖2-8所示,GetAddrInfoHandler最終的處理還是交由Bionic C的getaddrinfo函數來完成。根據前文所述,由于Netd進程設置了ANDROID_DNS_MODE環境變量,故Netd調用的getaddrinfo將走正常的流程。這個正常流程就是Netd進程將向指定的DNS服務器發起請求以解析域名。
Android系統中,通過這種方式來管理DNS的好處是所有解析后得到的DNS記錄都將緩存在Netd進程中,從而使這些信息成為了一個公共的資源,最大程度內做到了信息共享。

2.2.5 MDnsSdListener分析

MDnsSd是Multicast DNS Service Discovery的簡稱,它和Apple公司的Bonjour技術有關,故本節將先介紹Apple Bonjour技術。
1. Apple Bonjour技術介紹[4][5][6]
Bonjour是法語中的Hello之意。它是Apple公司為基于組播域名服務(multicast DNS)的開放性零配置網絡標準所起的名字。使用Bonjour的設備在網絡中自動組播它們自己的服務信息并監聽其他設備的服務信息,設備之間就像在打招呼,這也是該技術命名為Bonjour的原因。Bonjour使得局域網中的系統和服務即使在沒有網絡管理員的情況下也很容易被找到。
舉一個簡單的例子:在局域網中,如果要進行打印服務,就必須先知道打印服務器的IP地址。此IP地址一般由IT部門的人負責分配,然后他還得全員發郵件以公示此地址。有了Bonjour以后,打印服務器自己會依據零配置網絡標準在局域網內部找到一個可用的IP并注冊一個打印服務,名為“print service”之類的。當客戶端需要打印服務時,會先搜索網絡內部的打印服務器。由于不知道打印服務器的IP地址,客戶端只能根據諸如"print service"的名字去查找打印機。在Bonjour的幫助下,客戶端最終能找到這臺注冊了“print service”名字的打印機,并獲得它的IP地址以及端口號。
從Bonjour角度來看,該技術主要解決了三個問題:
Addressing:即為主機分配IP。Bonjour的Addressing處理比較簡單,即每個主機在網絡內部的地址可選范圍內找一個IP,然后查看下網絡內部是否有其他主機再用。如果該IP沒有被分配的話,它將使用此IP。
Naming:Naming解決的就是host和IP地址的對應關系。Bonjour采用的是Multiple DNS技術,即DNS查詢消息將通過UDP組播方式發送。一旦網絡內部某個機器發現查詢的機器名和自己設置的一樣,就回復這條請求。此外,Bonjour還拓展了MDNS的用途,即除了能查找host外,還支持對service的查找。不過,Bonjour的Naming有一個限制,即網絡內部不能有重名的host或service。
Service Discovery:SD基于上面的Naming工作,它使得應用程序能查找到網絡內部的服務,并解析該服務對應的IP地址和端口號。應用程序一旦得到服務的IP地址和端口號,就可以直接和該服務建立交互關系。
Bonjour技術在Mac OS以及Itunes、Iphone上都得到了廣泛應用。為了進一步推廣,Apple通過開源工程mdnsresponder將其開源出來。在Windows平臺上,它將生成一個后臺程序mdnsresponder。在Android平臺上(或者說支持POSIX的Linux平臺)它是一個名為mdnsd的程序。不過,不論是mdnsresponder還是mdnsd,應用開發者要做的僅僅是利用Bonjour的API向它們發起服務注冊、服務查詢和服務解析等請求并接收來自它們的處理結果。
下面我們將介紹Bonjour API中使用最多的三個函數,它們分別是服務注冊、服務查詢和服務解析。理解這三個函數的功能也是理解MDnsSdListener的基礎。
使用Bonjour API必須包含如下的頭文件和動態庫,并連接到:
#include <dns_sd.h> //必須包含此頭文件
libmdnssd.so //鏈接到此so
Bonjour中,服務注冊的API為DNSServiceRegister,原型如下:
DNSServiceErrorType DNSSD_API DNSServiceRegister
(
DNSServiceRef *sdRef,
DNSServiceFlags flags,
uint32_t interfaceIndex,
const char *name, /* may be NULL */
const char *regtype,
const char *domain, /* may be NULL */
const char *host, /* may be NULL */
uint16_t port, /* In network byte order */
uint16_t txtLen,
const void *txtRecord, /* may be NULL */
DNSServiceRegisterReply callBack, /* may be NULL */
void *context /* may be NULL */
);
該函數的解釋如下:
sdRef:代表一個未初始化的DNSService實體。其類型DNSServiceRef是指針。該參數最終由DNSServiceRegister函數分配內存并初始化。
flags:表示當網絡內部有重名服務時的沖突處理。默認是按順序修改服務名。例如要注冊的服務名為“printer”,當檢測到重名沖突時,就可改名為“printer(1)”。
interfaceIndex:表示該服務輸出到主機的哪些網絡接口上。值-1表示僅對本機支持,也就是該服務的用在loop接口上。
name:表示服務名,為空的話就取機器名。
regtype:服務類型,用字符串表達。Bonjour要求格式為"_服務名._傳輸協議",例如"_ftp._tcp"。目前傳輸協議僅支持TCP和UDP。
domian和host一般都為空。
port表示該服務的端口。如果為0的話,Bonjour會自動分配一個。
txtLen以及txtRecord字符串用來描述該服務。一般都設置為空。
callBack:設置回調函數。該服務注冊的請求結果都會通過它回調給客戶端。
context:上下文指針,由應用程序設置。
當客戶端需要搜索網絡內部特定服務時,需要使用DNSServiceBrowser API,其原型如下:
DNSServiceErrorType DNSSD_API DNSServiceBrowse
(
DNSServiceRef *sdRef,
DNSServiceFlags flags,
uint32_t interfaceIndex,
const char *regtype,
const char *domain, /* may be NULL */
DNSServiceBrowseReply callBack,
void *context /* may be NULL */
);
其中:
sdref、interfaceIndex、regtype、domain以及context含義與DNSServiceRegister一樣。
flags:在本函數中沒有作用。
callBack:為DNSServiceBrowser處理結果的回調通知接口。
當客戶端想獲得指定服務的IP和端口號時,需要使用DNSServiceResolve API,其原型如下:
DNSServiceErrorType DNSSD_API DNSServiceResolve
(
DNSServiceRef *sdRef,
DNSServiceFlags flags,
uint32_t interfaceIndex,
const char *name,
const char *regtype,
const char *domain,
DNSServiceResolveReply callBack,
void *context /* may be NULL */
);
其中:
name、regtype和domain都從DNSServiceBrowse函數的處理結果中獲得。
callBack用于通知DNSServiceResolve的處理結果。該回調函數將返回服務的IP地址和端口號。
2. MDnsSdListener分析
MDnsSdListener對應的Framework層服務為NsdService(Nsd為Network Service Discovery的縮寫),它是Android 4.1新增的一個Framework層Service。該服務的實現比較簡單,故本書不擬詳細討論它。感興趣的讀者不妨首先閱讀SDK中關于NsdService的相關文檔。
提示:SDK中有一個基于Nsd技術開發的NsdChat例程,讀者也可先學習它的實現。相關文檔位置為http://developer.android.com/training/connect-devices-wirelessly/nsd.html。
圖2-9所示為MDnsSdListener的家族成員示意圖。

圖2-9 MDnsSdListener家族成員
由圖2-9可知:
MDnsSdListener的內部類Monitor用于和mdnsd后臺進程通信,它將調用前面提到的Bonjour API。
Monitor內部針對每個DNSService都會建立一個Element對象,該對象通過Monitor的mHead指針保存在一個list中。
Handler是MDnsSdListener注冊的Command。
下面將簡單介紹MDnsSdListener的運行過程,主要工作可分成三步:
1)Netd創建MDnsSdListener對象,其內部會創建Monitor對象,而Monitor對象將啟動一個線程用于和mdnsd通信,并接收來自Handler的請求。
2)NsdService啟動完畢后將向MDnsSdListener發送"start-service"命令。
3)NsdService響應應用程序的請求,向MDnsSdListener發送其他命令,例如"discovery"等。Monitor將最終處理這些請求。
先來看第一步,當MDnsSdListener構造時,會創建一個Monitor對象,代碼如下所示:
[-->MDnsSdListener.cpp::Monitor:Monitor]
MDnsSdListener::Monitor::Monitor() {
mHead = NULL;
pthread_mutex_init(&mHeadMutex, NULL);
//創建兩個socket,用于接收MDnsSdListener對象的指令
socketpair(AF_LOCAL, SOCK_STREAM, 0, mCtrlSocketPair);
//創建線程,線程函數是threadStart,其內部會調用run
pthread_create(&mThread, NULL, MDnsSdListener::Monitor::threadStart, this);
}
Monitor的threadStart線程將調用其run函數,該函數通過poll方式偵聽包括mCtrlSocketPair在內的socket信息。這部分代碼屬于基本的Linux socket編程,本書不擬開展深入討論。
當NsdService發送"start-service"命令后,Handler的runCommand將執行Monitor的startService函數,代碼如下所示:
[-->MDnsSdListener.cpp::Monitor:startService]
int MDnsSdListener::Monitor::startService() {
int result = 0;
char property_value[PROPERTY_VALUE_MAX];
pthread_mutex_lock(&mHeadMutex);
//MDNS_SERVICE_STATUS是一個字符串,值為“init.svc.mdnsd”,在init.rc配置文件中,mdnsd是一個
//service,而“init.svc.mdnsd”將記錄mdnsd進程的運行狀態。
property_get(MDNS_SERVICE_STATUS, property_value, "");
if (strcmp("running", property_value) != 0) {
//如果mdnsd的狀態不為"running",則通過設置“ctl.start”命令啟動mdnsd
property_set("ctl.start", MDNS_SERVICE_NAME);
//如果mdnsd成功啟動,則屬性值變成"running"
wait_for_property(MDNS_SERVICE_STATUS, "running", 5);
result = -1;
} else {
result = 0;
}
pthread_mutex_unlock(&mHeadMutex);
return result;
}
startService的實現比較有趣,它充分利用了init的屬性控制以啟動mdnsd進程。
當NsdService發送注冊服務請求時,Handler的serviceRegister函數將被調用,代碼如下所示:
[-->MDnsSdListener.cpp::Handler:serviceRegister]
void MDnsSdListener::Handler::serviceRegister(SocketClient *cli, int requestId,
const char *interfaceName, const char *serviceName, const char *serviceType,
const char *domain, const char *host, int port, int txtLen, void *txtRecord) {
Context *context = new Context(requestId, mListener);
DNSServiceRef *ref = mMonitor->allocateServiceRef(requestId, context);
port = htons(port);
......
DNSServiceFlags nativeFlags = 0;
int interfaceInt = ifaceNameToI(interfaceName);
//調用Bonjour API DNSServiceRegister,并注冊回調函數MDnsSdListenerRegisterCallback
DNSServiceErrorType result = DNSServiceRegister(ref, interfaceInt,
nativeFlags, serviceName, serviceType, domain, host, port,
txtLen, txtRecord, &MDnsSdListenerRegisterCallback, context);
if (result != kDNSServiceErr_NoError) {
.....//錯誤處理
}
//通知Monitor對象進行rescan,請讀者自行研究該函數
mMonitor->startMonitoring(requestId);
cli->sendMsg(ResponseCode::CommandOkay, "serviceRegister started", false);
return;
}
DNSServiceRegister內部將把請求發送給mdnsd去處理,處理的結果通過MDnsSdListenerRegisterCallback返回,該函數代碼如下所示:
[-->MDnsSdListener.cpp::MDnsSdListenerRegisterCallback]
void MDnsSdListenerRegisterCallback(DNSServiceRef sdRef, DNSServiceFlags flags,
DNSServiceErrorType errorCode, const char *serviceName, const char *regType,
const char *domain, void *inContext) {
MDnsSdListener::Context *context =
einterpret_cast<MDnsSdListener::Context *>(inContext);
char *msg;
int refNumber = context->mRefNumber;
if (errorCode != kDNSServiceErr_NoError) {
......//錯誤處理
} else {
char *quotedServiceName = SocketClient::quoteArg(serviceName);
asprintf(&msg, "%d %s", refNumber, quotedServiceName);
free(quotedServiceName);
//將處理結果返回給NsdService
context->mListener->sendBroadcast(ResponseCode::ServiceRegistrationSucceeded,
msg,false);
}
free(msg);
}

提示 本節對Netd的工作流程進行了相關分析,這部分代碼相對簡單,處理流程也比較固定:
1)NM接收Kernel的UEvent消息,然后轉發給Framework層的客戶端。
2)CL、DPL以及MDnsSdListener接收來自客戶端的請求并處理它們。
唯一有趣的地方是Android中DNS的管理以及Apple Bonjour技術。感興趣的讀者不妨閱讀本章列出的參考資料以加深理解。

[1]此處結論來自bionic/libc/docs/OVERVIEW.txt文件,不過根據同目錄下CHANGES.txt的說明,resolv.conf將不再使用
====================================================================================
=========================略略略略略略略略略略略略略略==================================
2.5 本章總結和參考資料說明
2.5.1 本章總結
本章對Netd進行了詳細討論。相信讀者讀完此章的第一感受一定是代碼這么容易的模塊,竟然涉及如此多復雜的背景知識。確實,這也是專題卷所述內容的核心特點。從代碼上看也許它們并不復雜,但是其背后的理論知識卻可能大有來頭。對于這些內容而言,代碼只是外在的表現形式,其核心一定在其背后的那些知識中。所以,讀者在閱讀專題卷的時候,一定要考察自己是否對背景知識有所掌握。
概況而言,Netd涉及的內容和網絡管理與控制有關,例如DNS、Apple Bonjour、利用iptables等工具實現NAT、防火墻、帶寬控制、流量控制、路由控制功能,以及USB綁定Wi-Fi、SoftAP等。請讀者在本節的參考資料一覽中找到并繼續研究自己感興趣的內容。
最后,我們對NetworkManagementService進行了介紹。NMService的內容非常簡單。
2.5.2 參考資料說明
Linux PF_NETLINK相關資料
[1] Linux man PF_NETLINK
本文檔是Linux系統中的幫助文檔。從總體上介紹了PF_NETLINK(AF_NETLINK)的作用和相關的數據結構。對熟手比較適用。
[2] http://www.linuxjournal.com/article/8498
“Manipulating the Networking Environment Using RTNETLINK”,這篇文章以RTNETLINK為主要對象,介紹了如何利用它進行編程以操作網絡。此文寫得非常詳細,建議讀者深入閱讀,甚至自己動手寫測試例子。
DNS、Apple Bonjour相關資料
[3] http://baike.baidu.com/view/22276.htm
百度百科中關于dns的介紹,屬于入門級材料,不清楚的讀者可以先了解相關知識。
[4] http://en.wikipedia.org/wiki/MDNS
維基百科中關于Multicast DNS的介紹。入門級材料,但包含的信息不是很全,需要跟蹤其中的鏈接才能對MDNS有全面了解。
[5] https://developer.apple.com/library/mac/#documentation/Cocoa/Conceptual/NetServices/Introduction.html#//apple_ref/doc/uid/TP40002445-SW1
“Introduction to Bonjour Overview”,蘋果開發網站上關于Bonjour基礎知識的入口,包含“About Bonjour”、“Bonjour API Architecture”等文檔。
[6] https://developer.apple.com/library/mac/#documentation/Networking/Conceptual/dns_discovery_api/Introduction.html#//apple_ref/doc/uid/TP30000964
“DNS Service Discovery Programming Guide”,蘋果開發網站關于NSD API的說明。
iptables相關資料
iptables的相關文檔非常多,雖然Linux也提供了幫助文檔(man iptables),但對新手來說該文檔實在不是學習的好資料。
[7] http://www.thegeekstuff.com/2011/01/iptables-fundamentals/
“Linux Firewall Tutorial: IPTables Tables, Chains, Rules Fundamentals”,這篇文章首先從原理上介紹了如何去理解iptables,然后介紹了相關的例子。筆者認為它是iptables最好的入門資料。
[8] http://selboo.com.cn/post/721/
“iptables的相關概念和數據包的流程”,這篇文檔介紹了iptables中各個table及chain的處理順序,請讀者結合[7]來理解iptables。
[9] http://www.frozentux.net/iptables-tutorial/cn/iptables-tutorial-cn-1.1.19.html
“Iptables 指南 1.1.19”,這篇文檔介紹的iptables版本比較舊(Android 4.2使用的iptables版本是1.4.11),但對iptables常用參數都有非常詳細的介紹。適合入門后的讀者進行深入閱讀。
TC相關資料
tc文獻的數量和難度遠大于iptables,此處精選幾個必讀文獻。
[10] http://linux-ip.net/articles/Traffic-Control-HOWTO/intro.html
“Traffic Control HOWTO”,理解traffic control的必讀文獻,覆蓋面很全,理論知識講解到位。難度稍大,需要仔細琢磨才能完全理解。
[11] http://wenku.baidu.com/view/f02078db50e2524de5187e45.html
“TC(Linux下流量控制工具)詳細說明及應用實例”,百度文庫中的一篇文檔,篇幅雖然不長,但也做到了理論和實例結合。建議讀者先閱讀此文獻,然后再深入研究[10]。
[12] http://fanqiang.chinaunix.net/a1/b1/20010811/0705001103.html
“在LINUX中實現流量控制器”,介紹TC的一篇博文,主要對tc的命令用法列舉了不少實例,屬于tc的實戰文章。建議放到最后閱讀。
[13] http://www.linuxfoundation.org/collaborate/workgroups/networking/ifb
這是筆者能找到的關于IFB設備最完整的資料,對IFB的使用、常規用法等進行了全方位的介紹。
IP命令相關資料
ip命令比較簡單,這里僅給出一篇文獻。
[14] http://blog.chinaunix.net/uid-24921475-id-2547198.html
Linux ip命令介紹
NetDevice編程文獻
[15] Linux man netdevice
非常詳細的NetDevice編程介紹,建議讀者認真閱讀。
Linux策略路由相關資料
[16] http://www.cnblogs.com/iceocean/articles/1594488.html
“Linux策略路由”,中文文檔,知識面覆蓋較全,屬于入門級資料。
[17] http://www.policyrouting.org/PolicyRoutingBook/ONLINE/TOC.html
“Policy Routing With Linux”,這是一本完整的書籍(可見網管是一個復雜的工作)。個人感覺[16]是參考[17]的學習總結。屬于高級閱讀材料,難度較大。
Linux IPv6控制相關資料
[18] http://www.ipsidixit.net/2012/08/09/ipv6-temporary-addresses-and-privacy-extensions/
“IPv6 temporary addresses and privacy extensions”,介紹Linux中IPv6臨時地址和privacy extensions方面的知識,知識覆蓋面較全。屬于入門資料。

TTY和ptmx編程相關資料
[19] http://tldp.org/HOWTO/Text-Terminal-HOWTO.html
“Text-Terminal-HOWTO”,比較舊的資料,覆蓋面非常廣。讀者可僅閱讀自己想了解的章節。
[20] http://blog.tianya.cn/blogger/post_read.asp?BlogID=3616841&PostID=33399981
“Linux下tty/pty/pts/ptmx 詳解”,中文寫的好材料,還列出了其參考的文獻。最后,關于ptmx,讀者還可通過man ptmx獲得如何用它進行編程的指導。
PPP和Pppd相關資料
[21] http://tldp.org/HOWTO/PPP-HOWTO/
“Linux PPP HOWTO”,Linux HowTo系列的內容都簡單易懂。雖章節較多,但很多內容僅一兩句了事。可做入門參考。
[22] http://network.51cto.com/art/201009/223784.htm
“基礎解讀PPP協議”,中文文檔,一頁內容,主要介紹PPP框架性的內容。
[23] http://wenku.baidu.com/view/0c395f15866fb84ae45c8d4a.html
“ppp介紹”,百度文庫中的一個關于ppp的PPT。內容翔實,不僅介紹了ppp協議的數據包,也從框架上介紹了ppp的工作流程。建議讀者首先閱讀此文獻。
[24] Linux man pppd
介紹pppd中各個選項的作用。
NAT相關資料
[25] http://oa.jmu.edu.cn/netoa/libq/pubdisc.nsf/66175841be38919248256e35005f4497/7762e8e1056be98f48256e88001ef71d?OpenDocument
“用iptables實現NAT”,中文文檔,簡單易懂。
Tether、RNDIS、DHCP、DNSmasq相關資料
[26] http://en.wikipedia.org/wiki/Tethering
“Tethering”,維基百科中關于Tether的介紹,淺顯易懂,屬于普及型資料。
[27] http://msdn.microsoft.com/en-us/library/windows/hardware/gg463293.aspx
“Remote NDIS (RNDIS) and Windows”,MSDN文檔,非常翔實(不得不說微軟在文檔方面的工作真的是一絲不茍)。
[28] http://baike.baidu.com/view/7992.htm?subLemmaId=7992&fromenter=%A3%C4%A3%C8%A3%C3%A3%D0
百度百科中關于DHCP的解釋,入門資料。
[29] http://baike.baidu.com/view/6681631.htm
百度百科中關于DNSmasq的解釋。
[30] http://wenku.baidu.com/view/662b536b561252d380eb6ec1.html
關于DHCP協議中option字段的詳細介紹。
Softap和hostapd相關資料
[31] 《802.11 無線網絡權威指南中文第二版》
讀者可先閱讀第1、2章中關于Wi-Fi技術中的一些基本概念,例如AP和Station。
[32] http://baike.baidu.com/view/2475889.htm
百度百科關于SoftAp的入門級介紹。
[33] 關于hostapd,讀者可利用man hostapd得到各個選項的用法。
提示,讀者必須先安裝hostapd,然后才能查閱其幫助文檔。

總結

以上是生活随笔為你收集整理的《深入理解Android:Wi-Fi,NFC和GPS》章节连载[节选]--第二章 深入理解Netd的全部內容,希望文章能夠幫你解決所遇到的問題。

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

色综合久久久网 | 狂野欧美激情性xxxx | 一区二区三区四区五区六区 | 久久久久久久毛片 | 日韩高清在线不卡 | 亚洲欧美视频网站 | 国产剧情在线一区 | 免费av观看 | 国产福利一区二区三区在线观看 | 中文字幕一区二区三区视频 | 国产在线a | 一本一本久久a久久精品综合妖精 | 婷婷综合导航 | 色综合久久66| 国产激情久久久 | avav片| 久久伊人综合 | 91av在线播放视频 | 欧美激情精品久久久久久变态 | 久久精品一区二区三区国产主播 | 开心激情网五月天 | 91久久电影 | 久草在线视频在线观看 | 成人av影院在线观看 | 欧美色道 | 亚洲欧洲日韩 | 九九色综合 | 国产精品久久久久久久久久久久午 | 特黄特色特刺激视频免费播放 | 8x成人免费视频 | 国产区精品区 | 亚洲国产精品成人综合 | 久久精品人人做人人综合老师 | 99久久精品国产一区 | 黄色国产高清 | 色99久久 | 一区二区中文字幕在线播放 | 国产成人精品久久 | 美女久久一区 | 99麻豆久久久国产精品免费 | 国产精品99久久久久久有的能看 | 欧美日本高清视频 | 久久精品久久久久久久 | 国产精品av在线免费观看 | 日日射天天射 | 热久久视久久精品18亚洲精品 | 色综合天天色综合 | 国产精品18久久久久久首页狼 | 精品国产乱码久久久久久天美 | 日韩精品中文字幕在线观看 | 91在线中字 | 亚洲一区二区三区四区在线视频 | 色婷婷综合视频在线观看 | 超碰人人做 | 麻豆91在线 | 久久久五月天 | 国产精品手机在线播放 | 亚洲精品在线观看视频 | 欧美天天综合 | 日日麻批40分钟视频免费观看 | 欧美三级免费 | 日韩二区在线播放 | 中文字幕日本在线观看 | 天天色天天操天天爽 | 国产啊v在线观看 | 在线免费看黄色 | 91黄色影视 | 免费观看9x视频网站在线观看 | 五月天婷婷综合 | 国产精品久久久久久爽爽爽 | 久久免费国产精品1 | 天天操综合网站 | 日韩精品一区二区三区在线播放 | 久久久久久久久久久免费视频 | 精品视频免费在线 | 天天操天天干天天摸 | a久久免费视频 | av在线免费观看网站 | 国产最新视频在线 | 伊人亚洲综合 | 91亚洲欧美激情 | 黄色小说视频在线 | 午夜国产福利在线观看 | 日韩欧美视频一区二区 | 狠狠操在线 | 四虎在线观看 | 日日夜夜噜 | 久久久首页 | 免费福利在线观看 | 久草视频在线免费看 | 天天综合久久综合 | 国产黄色视 | 亚洲一二区视频 | 日韩免费电影在线观看 | 国产精品亚洲人在线观看 | 欧美精品在线一区二区 | 欧美亚洲免费在线一区 | 免费看一级特黄a大片 | 国内精品在线看 | 91精品免费视频 | 日韩二区精品 | 婷婷日韩| 夜夜操天天操 | 日女人免费视频 | 亚洲久久视频 | 超碰人人超| 色播六月天 | 人人精久 | 国产美女主播精品一区二区三区 | 成人在线免费观看视视频 | 97超视频免费观看 | 久久99精品久久久久久秒播蜜臀 | 激情欧美xxxx | 国产美女主播精品一区二区三区 | 国产黄色片久久 | 超级碰视频| 久久免费av电影 | 在线观看麻豆av | 欧洲视频一区 | 婷婷色av | 久久久久免费精品国产小说色大师 | 国产黄大片在线观看 | 中文在线免费一区三区 | 免费日韩 精品中文字幕视频在线 | 亚洲精品系列 | 亚洲成免费 | 91人人视频在线观看 | 免费又黄又爽的视频 | 日批视频在线观看免费 | 欧美先锋影音 | 在线观看网站你懂的 | av中文字幕在线观看网站 | 国际精品久久 | 丝袜美腿在线 | 国产成人资源 | 黄色在线视频网址 | 丁香六月婷婷激情 | 中文字幕色综合网 | 国产 视频 久久 | 国产精品96久久久久久吹潮 | 欧美日韩视频 | 久久国产色 | 高清av网站| 亚洲第一av在线播放 | 亚洲aⅴ在线观看 | 日韩精品在线观看av | 欧美日韩成人一区 | 在线色亚洲 | 国产精品久久一区二区三区, | 丝袜精品视频 | 久久电影日韩 | 日韩av男人的天堂 | 免费观看www小视频的软件 | 久久久蜜桃一区二区 | 中文字幕丝袜一区二区 | av中文在线 | 欧美色插 | 日韩欧美高清视频在线观看 | 超碰97在线看 | 成人黄大片 | 天天综合色天天综合 | 国产精品一区二区白浆 | 国产一区二区综合 | 欧美一级性 | 中文字幕在线视频国产 | 蜜臀av一区二区 | 成人av免费播放 | 在线观看国产日韩 | 激情综合网天天干 | 国产精品专区h在线观看 | 香蕉视频在线免费看 | 99久久婷婷国产一区二区三区 | 天天av在线播放 | 精品产品国产在线不卡 | 6080yy精品一区二区三区 | 中文字幕免费国产精品 | 久久91久久久久麻豆精品 | 欧美日韩三区二区 | 亚洲午夜精品久久久久久久久 | 五月婷久久 | 国产精品视频 | 日本最新中文字幕 | 中文字幕色在线 | 亚洲精品福利在线 | 国产高清免费在线播放 | 不卡中文字幕在线 | 国产精品久久久毛片 | 一区二区三区四区影院 | 成人a在线观看高清电影 | 午夜色大片在线观看 | www.com久久久 | 国内精品久久久久国产 | 婷婷av网站 | 天天干天天操av | 91久久精品日日躁夜夜躁国产 | 超碰在线人 | 五月天伊人 | 区一区二在线 | 亚洲无线视频 | 欧美一区二区三区免费观看 | 99在线高清视频在线播放 | 国产福利电影网址 | 婷婷5月色 | 在线中文字幕视频 | 国产视频 久久久 | 日韩免费不卡视频 | av在线免费播放网站 | 成年人视频在线免费观看 | 2024国产精品视频 | 国产在线看 | 国产91影视 | 日本中文乱码卡一卡二新区 | 亚洲成人黄色网址 | 欧美91av| 在线观看免费黄色 | 国产黄a三级三级 | 亚洲天堂网视频在线观看 | 欧美亚洲另类在线视频 | 在线 视频 亚洲 | 精品福利视频在线观看 | 黄色亚洲 | 亚洲精品国产自产拍在线观看 | 亚洲激色 | 久久久99精品免费观看乱色 | 91中文在线| 成人午夜影院在线观看 | 欧美日韩中字 | 人人干干人人 | 欧美国产不卡 | 青青河边草免费视频 | 国产拍揄自揄精品视频麻豆 | 亚洲第一区在线播放 | 四虎永久免费网站 | 91视频 - 114av | 人人玩人人添人人澡97 | 不卡av在线免费观看 | 全黄色一级片 | 狠狠ri | 99视频导航 | av动图| 国产精品久久久久毛片大屁完整版 | 亚洲最新毛片 | 亚洲三级黄色 | 去看片 | 成人动态视频 | 欧美性超爽| 中文字幕在线不卡国产视频 | 91网站在线视频 | 国产精品久久久久999 | 亚洲精品国产综合99久久夜夜嗨 | 日韩精品一区二区在线观看 | 欧美日韩三级 | 综合久久网站 | v片在线看 | 久久精品美女视频网站 | 久久久久免费精品视频 | 在线色亚洲 | 福利网在线 | 国产91成人在在线播放 | 国产精品色视频 | 国内精品视频在线播放 | 在线影院av | 岛国片在线 | 热99在线视频 | 日韩精品字幕 | 精品国产人成亚洲区 | 亚洲伦理电影在线 | 91福利视频网站 | 色在线视频 | 国产高清中文字幕 | 91麻豆精品一区二区三区 | 精品久久91 | 精品在线播放 | 91字幕| 九九99| 在线中文字幕一区二区 | 成人性生交大片免费观看网站 | 久久午夜羞羞影院 | 天天综合色 | 国产日韩三级 | 免费h漫在线观看 | 中文资源在线官网 | 香蕉视频在线观看免费 | 一级黄色片在线播放 | 黄污网站在线观看 | 免费在线观看不卡av | 国产精品欧美久久 | 色综合网在线 | 欧美成人黄色片 | 黄色亚洲免费 | 国产成人精品亚洲精品 | 国产日产精品久久久久快鸭 | 国产网红在线观看 | 在线视频成人 | 操操操日日 | 婷婷激情久久 | 欧美激情综合五月色丁香小说 | 天天操综 | 九色视频网站 | 福利视频一区二区 | 精品国产一区二区三区四区vr | 久草在线最新免费 | 中文字幕丝袜美腿 | 日韩精品在线观看av | 久久综合在线 | 欧美analxxxx | 麻豆传媒视频在线 | 成人午夜在线观看 | av不卡免费在线观看 | 午夜视频在线瓜伦 | 免费进去里的视频 | 日韩在线视频观看免费 | 偷拍精偷拍精品欧洲亚洲网站 | 在线观看视频你懂 | www操操| 日韩av成人在线观看 | 91自拍视频在线观看 | 亚洲做受高潮欧美裸体 | 美女黄色网在线播放 | 免费色av | 热久久国产精品 | 中文字幕色网站 | 欧美日韩免费在线观看视频 | 国产女教师精品久久av | 久久国产免费看 | 五月婷婷激情五月 | 国产护士av| 99热最新地址 | 97国产小视频 | 99精品亚洲| 国产精品一区二区久久精品 | 国产精品尤物视频 | 久久午夜色播影院免费高清 | 9在线观看免费高清完整版在线观看明 | 欧美在线视频一区二区三区 | 久久av电影| 在线视频99 | 日韩av中文在线观看 | 国内精品久久久久久中文字幕 | 久久亚洲人 | 国产精品久久久久久久久久三级 | 91cn国产在线 | 午夜少妇 | 欧美在线91 | 日韩a级免费视频 | 一区二区三区免费看 | 美国人与动物xxxx | 国产精品乱码一区二区视频 | 欧美综合干 | 亚洲狠狠丁香婷婷综合久久久 | 狠狠狠色丁香婷婷综合久久88 | 日韩av免费在线电影 | 91污污| 97超碰在线久草超碰在线观看 | 免费福利在线观看 | 日韩免费一区二区在线观看 | 亚洲精品18日本一区app | 亚洲精品久久视频 | 国产精品麻豆免费版 | www.夜夜草 | 中文字幕一区在线观看视频 | 91大神精品视频 | 性色av一区二区三区在线观看 | 激情综合交| 久久国产精品色婷婷 | 国产精品免费人成网站 | 2000xxx影视 | 精品亚洲男同gayvideo网站 | 国产精品18久久久久久久 | 免费观看完整版无人区 | 久久综合色播五月 | 国内精品久久久久久久久久清纯 | av免费成人| 亚洲精品玖玖玖av在线看 | 午夜在线免费观看 | 中文字幕在线观看第二页 | 国产精品v欧美精品v日韩 | 日韩av片免费在线观看 | 日本九九视频 | 国内精品久久久久影院一蜜桃 | www.av中文字幕.com| 在线免费精品视频 | 麻豆一区二区三区视频 | 视频一区视频二区在线观看 | 91视频在线免费看 | 中文高清av | 在线视频免费观看 | 日韩av不卡在线 | www.午夜色.com | 久久精视频 | 3d黄动漫免费看 | 亚洲国产精品影院 | 91看毛片| www.91成人 | 国产一级性生活 | 亚洲黄色三级 | 精品久久久久一区二区国产 | 蜜臀av夜夜澡人人爽人人桃色 | 最近高清中文字幕在线国语5 | 国产精品久久久久久999 | 中文字幕免 | 国产小视频在线看 | 国产美女免费看 | 国产精品2区 | av千婊在线免费观看 | 四虎在线视频免费观看 | 日韩黄视频 | 欧美成人中文字幕 | 高清av中文在线字幕观看1 | 久久视频这里有久久精品视频11 | 国产日韩欧美精品在线观看 | 亚洲精品国产综合99久久夜夜嗨 | 婷婷丁香色综合狠狠色 | 国产午夜精品福利视频 | 麻豆视频在线 | 日韩视频在线观看视频 | 国产四虎在线 | 国产精品综合久久久久 | 国产精品原创 | 久久亚洲人 | 国产日韩中文字幕 | 日韩精品一区电影 | 亚洲成a人片77777潘金莲 | 国内视频一区二区 | a资源在线 | 色天天天| 国产精品黑丝在线观看 | 亚洲精品在线观看的 | 99久久视频| 日韩精品一区二区三区水蜜桃 | 国产精品一区二区麻豆 | 欧美一区二区在线 | 国产又粗又长又硬免费视频 | 日本公妇色中文字幕 | 午夜国产在线 | 在线观看一区 | 亚洲情感电影大片 | 免费高清在线一区 | 精品久久久国产 | 色视频网址 | www.午夜视频 | 久久久久97国产 | 中文字幕在线成人 | 久久免费视频一区 | 亚洲精品美女久久久久网站 | 久久不见久久见免费影院 | 日韩一二三区不卡 | 97成人精品视频在线观看 | 久久都是精品 | 91福利视频免费观看 | 久久av一区二区三区亚洲 | 在线免费91| 精品国产精品久久一区免费式 | 黄色成人毛片 | 欧美另类69 | 久久黄色免费观看 | 精品国产精品国产偷麻豆 | 亚洲精品久久久久999中文字幕 | 中文字幕日本在线 | 伊人精品影院 | 中文字幕成人一区 | av网站在线观看免费 | 午夜婷婷在线播放 | 国产一区高清在线观看 | 成年人在线观看免费视频 | 成人a级黄色片 | 久久免费99精品久久久久久 | 国产精品毛片久久久久久久 | 免费精品视频在线 | 探花视频免费在线观看 | 99色免费 | 亚洲国产精品va在线看黑人 | 日韩精品一区在线观看 | 精品福利在线视频 | av中文字幕网站 | 97超碰在线免费 | 综合伊人av | 国产精品乱看 | 国产视频综合在线 | 丁香六月久久综合狠狠色 | 丁香花在线观看免费完整版视频 | 丁香五月网久久综合 | 国产99久久九九精品免费 | 亚洲黄网址| 91在线视频免费观看 | 97国产精品一区二区 | 一区二区三区免费在线 | 欧美91精品国产自产 | 综合色天天 | 国产精品成人一区二区三区吃奶 | 亚洲国产偷 | 久久99热这里只有精品 | 国产中文字幕视频在线 | 麻豆91小视频 | 免费在线观看成人 | 91亚洲精品在线观看 | 国产 在线 日韩 | 91av视频播放| 中文字幕在线资源 | 日韩欧美视频在线免费观看 | 看片一区二区三区 | 精品美女在线观看 | 欧美日韩一区二区久久 | 日韩视频在线观看免费 | 一本一本久久a久久精品综合小说 | 在线免费视频a | 精品国产精品国产偷麻豆 | 一区二区三区在线视频111 | 热热热热热色 | 毛片a级片 | 国产色妞影院wwwxxx | 最近的中文字幕大全免费版 | av中文在线观看 | 国产精品毛片一区视频 | 国产视频每日更新 | 精品久久久久一区二区国产 | 久久黄网站 | www.亚洲视频| 日韩中文在线视频 | 亚洲精品美女久久 | 亚洲乱码久久久 | 91中文字幕在线观看 | 97香蕉久久超级碰碰高清版 | 亚洲成人免费 | 少妇bbw搡bbbb搡bbb | 成人免费毛片aaaaaa片 | 一区二区三区在线不卡 | 在线精品亚洲 | 欧美日韩一区二区视频在线观看 | 成人三级av | 在线国产欧美 | 天天综合在线观看 | 国产97色在线 | 国产免费嫩草影院 | 国产男女爽爽爽免费视频 | 成人av高清在线观看 | 国产手机在线 | 亚州欧美精品 | 国内精品视频在线 | 五月综合婷 | 五月婷婷综合在线观看 | 欧美精品一级视频 | 亚洲精品影视 | 国产在线不卡视频 | 久久精品电影院 | 婷婷激情站 | 久久天| 亚洲视频在线播放 | 在线观看免费观看在线91 | 久操97| 免费av试看 | 91桃色国产在线播放 | 国产91丝袜在线播放动漫 | 天天摸天天操天天爽 | 国产成人精品久久久 | 欧美日韩综合在线 | 久久99网 | 亚洲精品一区二区在线观看 | 亚洲欧美少妇 | 国产精品一区二区你懂的 | 亚洲一区精品二人人爽久久 | 黄色影院在线免费观看 | 久久国产精品久久精品国产演员表 | 欧美网站黄色 | 亚洲砖区区免费 | 国产午夜精品一区二区三区在线观看 | 伊人色播 | 国内丰满少妇猛烈精品播放 | 成人中文字幕在线 | 久久五月激情 | 中文字幕在线日亚洲9 | 久久视频免费在线观看 | 午夜国产成人 | 欧美日韩中文在线视频 | 国产经典 欧美精品 | 久久久久9999亚洲精品 | 狠狠色狠狠色综合日日小说 | 国产福利在线不卡 | 亚州精品天堂中文字幕 | 一区二区网 | 国产一二区在线观看 | a精品视频| 亚洲一区二区三区在线看 | 蜜桃av人人夜夜澡人人爽 | 国产涩涩网站 | 在线看片成人 | 丁香导航 | 久久r精品 | 国产99久久久欧美黑人 | 亚洲视频免费在线观看 | 91手机电影 | 韩国av免费在线 | 久久美女免费视频 | 久久精彩| 9在线观看免费高清完整 | 特黄色大片 | 免费看国产精品 | 在线观看小视频 | 国产精品99久久久久人中文网介绍 | 在线国产中文字幕 | 热99在线视频 | av黄色免费看 | 91亚洲精品久久久蜜桃网站 | 日韩有码中文字幕在线 | 在线免费国产 | 91福利国产在线观看 | 婷婷六月天丁香 | 久视频在线播放 | 在线国产小视频 | 天天综合五月天 | 欧美精品久久久久久久 | 日韩国产欧美在线视频 | 欧美日韩在线视频一区 | 91视频88av | 狠狠插天天干 | 狠狠干激情 | 欧美三级高清 | 久久,天天综合 | 国产91亚洲 | 日韩一区二区免费在线观看 | 色婷婷狠狠 | 日韩久久久久久久久久久久 | 国产精品一区在线观看你懂的 | 黄色国产高清 | 国产成人三级三级三级97 | 综合激情av | 99精品福利 | 毛片一区二区 | 精品国产a | 人人看黄色 | 国产不卡一二三区 | 日本视频网| 九九九九热精品免费视频点播观看 | 99夜色| 日本韩国精品在线 | 国产亚洲精品v | 婷婷在线视频观看 | 天天鲁天天干天天射 | 成人av免费在线观看 | 五月天久久 | 亚洲国产理论片 | 91av在线电影 | 久久人人爽人人爽人人片av免费 | 亚洲dvd| 国产成人一区二区精品非洲 | 久久久久亚洲天堂 | 国产一区二区在线免费观看 | 成人av视屏| 欧美精品一区二区性色 | 久久艹国产 | 国产精品99精品久久免费 | 天天综合视频在线观看 | 亚洲涩涩网 | 日韩中文字幕免费看 | 嫩草av在线 | 国产理论一区二区三区 | 91亚洲精品久久久中文字幕 | 中文在线最新版天堂 | 黄色电影小说 | 天天干天天草天天爽 | 嫩小bbbb摸bbb摸bbb | 男女精品久久 | 欧美性色综合网站 | 中文字幕av在线免费 | 一区在线电影 | 国内久久久久久 | 国产淫片 | 亚洲午夜精品电影 | av高清在线观看 | 麻花天美星空视频 | 亚洲欧美日本国产 | 日韩影片在线观看 | 91一区一区三区 | 天天天天天天天天操 | 91视频电影 | 青草视频在线 | 精品国产aⅴ麻豆 | 久av在线| 久久99久国产精品黄毛片入口 | 国产麻豆剧果冻传媒视频播放量 | 最新日韩视频在线观看 | 91精品办公室少妇高潮对白 | 好看av在线 | 国产精品色在线 | 国产精品2020 | 亚洲影院国产 | 国产午夜小视频 | av韩国在线 | 日韩和的一区二在线 | 欧美精品久久久久久久久久久 | 国内精品久久久久久久久 | 国产精品久久久久一区二区国产 | 中国美女一级看片 | av免费成人 | 免费的国产精品 | 五月婷婷丁香色 | 国产精品av在线 | 91亚州| 91精品国产高清自在线观看 | 日韩a免费 | 亚洲四虎在线 | 欧美精品资源 | 国产午夜精品一区二区三区 | 在线欧美最极品的av | 亚洲视频综合 | 国产精品一级视频 | 久久精品一二三 | 在线一区观看 | 2019免费中文字幕 | av不卡中文| 久久精品国产成人精品 | 久久久黄视频 | 99久热在线精品视频 | 精品夜夜嗨av一区二区三区 | 天天操天天舔天天干 | 欧美日韩精品区 | 日韩免费av网址 | 久久综合色婷婷 | 婷婷丁香久久五月婷婷 | 欧美日韩高清不卡 | 精品国产一区二区三区在线 | 国产精品久久久久av福利动漫 | 天天色天 | 四虎国产精品免费 | 免费视频a| 人人超碰免费 | 99 国产精品| 欧美整片sss | 精品免费一区 | 国模视频一区二区 | 91av视频在线免费观看 | 91丨九色丨高潮 | 九九精品在线观看 | 亚洲欧美国产日韩在线观看 | 中文字幕电影高清在线观看 | 一级黄色片毛片 | 免费视频网 | 婷婷五情天综123 | 久久久久久久久艹 | 偷拍视频一区 | 亚洲精品456在线播放第一页 | 日韩欧美高清在线观看 | 国产一区在线播放 | 婷婷激情在线观看 | 在线播放亚洲激情 | av电影免费 | 97视频播放 | 亚洲综合色视频在线观看 | 91黄色免费看 | 久久精品男人的天堂 | 免费a视频 | 亚洲一区网 | 亚洲人成综合 | av资源网在线播放 | 久久久久99精品成人片三人毛片 | 日韩理论电影在线观看 | 久久伊人91 | 日韩精品视频网站 | 在线日本看片免费人成视久网 | 在线观看国产麻豆 | 日韩久久久久久久久久久久 | 日韩免费三级 | 黄色av网站在线免费观看 | 91久草视频| 五月开心婷婷 | 91亚洲免费 | 色五月激情五月 | 日本精品视频在线 | 国产亚洲精品久久久久久电影 | 国产精品2020 | 91精品国产三级a在线观看 | 日韩精品在线看 | 国产欧美日韩一区 | 就操操久久 | 91干干干 | 日韩精品一区二区三区电影 | 国内精品久久久久久久影视简单 | 国产你懂的在线 | 国产做a爱一级久久 | 中文字幕综合在线 | 一本一道久久a久久综合蜜桃 | 夜夜操夜夜干 | 成人av免费看 | 久草精品视频在线看网站免费 | 超碰97中文| 日韩,精品电影 | 国产丝袜在线 | 二区三区中文字幕 | 日韩中文字幕免费在线观看 | 亚洲黄色免费在线看 | 婷婷婷国产在线视频 | 久久久在线免费观看 | 麻豆成人在线观看 | 激情综合狠狠 | 五月天综合网站 | 国产亚洲片| 91亚洲影院 | 精品毛片久久久久久 | 精品久久久精品 | 久久第四色 | 欧美日韩精品影院 | 正在播放亚洲精品 | 精品免费国产一区二区三区四区 | 波多野结衣视频一区二区 | 亚洲午夜精品久久久久久久久久久久 | a级成人毛片| 中文字幕123区 | 免费av 在线| 国产精选在线观看 | 在线日韩亚洲 | 国产精品美 | 99 精品 在线 | 九九在线高清精品视频 | 日韩黄色在线电影 | 激情婷婷av | 国产91精品久久久久 | 最近中文字幕在线 | 国产性xxxx | 92精品国产成人观看免费 | 色多多污污在线观看 | 国产成人精品三级 | 国产一级一级国产 | 午夜视频免费 | 中文字幕在线看视频 | 水蜜桃亚洲一二三四在线 | 深夜国产福利 | 91在线在线观看 | 国产精选在线观看 | 亚洲精选视频在线 | 久久综合导航 | 亚洲成年人免费网站 | 亚洲一区二区观看 | 国产精品国产精品 | 在线播放 亚洲 | 久久久精品一区二区三区 | 久久美女视频 | 久久久久久久综合色一本 | 777视频在线观看 | 四虎影视8848aamm | 日本中文在线观看 | 中文字幕日本特黄aa毛片 | 精品美女久久久久久免费 | 久久 精品一区 | 1000部18岁以下禁看视频 | 97国产人人 | 亚洲二级片| 一区二区亚洲精品 | 911久久| 欧美成人免费在线 | 免费网址你懂的 | 激情欧美在线观看 | 国产一区二区高清不卡 | 中文字幕在线观看2018 | 91久久国产露脸精品国产闺蜜 | 看毛片网站 | 色香蕉在线视频 | 日日夜夜狠狠 | 97在线免费视频观看 | av资源在线观看 | 亚洲一二三久久 | 亚洲国产资源 | 日韩电影中文字幕在线 | 国产一线二线三线在线观看 | 免费精品久久久 | 国产xxxx| 在线观看日本高清mv视频 | 狠狠ri | 成人国产精品免费 | 超碰在线cao| 91精品国产一区二区三区 | 国产成人精品a | 69性欧美| 国产群p视频 | 成人超碰在线 | 人人爽人人舔 | 日日夜夜精品网站 | 日韩丝袜视频 | 成人av电影免费在线观看 | 国产精品免费久久 | 日韩在线一级 | 免费三级a| 成年人免费在线观看网站 | 91精品第一页 | 久草影视在线 | 国产黄色精品在线观看 | 美女精品在线 | 欧美综合在线视频 | 99久久精品网 | 久久国产视频网 | 日韩区在线观看 | 美女网站色 | 成人免费av电影 | 高清不卡一区二区在线 | 日韩伦理片hd | 欧美午夜一区二区福利视频 | 五月婷婷视频 | 婷婷视频在线播放 | 亚洲精品乱码久久久久久蜜桃不爽 | 麻豆一精品传二传媒短视频 | 成人国产精品入口 | 天天干天天射天天操 | 91成人网在线播放 | 好看的国产精品视频 | 99免费在线 | 亚洲天堂网在线播放 | 亚洲片在线资源 | 久久精品五月 | 欧美成年网站 | 久久人人插 | 黄色美女免费网站 | 99久高清在线观看视频99精品热在线观看视频 | 9999在线 | 国产精品麻豆99久久久久久 | 亚洲成免费 | 久久艹免费 | 激情丁香月 | 91av综合| 婷婷中文在线 | 亚洲欧美国产精品va在线观看 | 欧美久久久久久久久久久 | 亚洲人成在线观看 | 久久影院午夜论 | 久久成人国产精品 | 亚洲成av人片在线观看香蕉 | 亚洲精品国产精品国产 | 国产精品手机看片 | 人人澡人人草 | 91电影福利 | 亚洲精品国产欧美在线观看 | 精品日韩视频 | 在线看片视频 | 国产91影院| 91尤物国产尤物福利在线播放 | 青青久视频| 这里只有精品视频在线观看 | 三级黄在线 | 97国产精品 | 亚洲激情视频在线观看 | 免费av影视 | 亚州视频在线 | 久久在线观看视频 | 国产乱码精品一区二区三区介绍 | 亚洲精品视频免费在线观看 | 中文字幕精品一区久久久久 | 免费在线成人av电影 | 日韩大陆欧美高清视频区 | 久久精品资源 | av直接看 | 国产中文字幕av | 国产成人精品免高潮在线观看 | 国产成人亚洲在线观看 | 色综合咪咪久久网 | 高清av免费一区中文字幕 | av激情五月 | www成人精品 | 成人久久精品视频 | 狠狠操狠狠操 | 日本久久91 | 日韩欧美高清一区二区三区 | 国产精品99久久久久久人免费 | 2019久久精品| 麻花豆传媒mv在线观看网站 | 国产九九九精品视频 | 精品色999| 人人干在线观看 | 国产一区二区精 | 五月色综合 | 欧美成人视 | 在线观看视频91 | 精品福利网站 | 成年人精品 | 在线免费黄色毛片 | 亚洲电影网站 | 国产69精品久久久久久 | 91成年人视频 | 热久在线 | 国产精品1区2区 | 日韩在线视频国产 | 国产精品久久99综合免费观看尤物 | 国产免费午夜 | 精品国产成人在线 | 丁香婷婷综合五月 | 91在线入口 | 精品免费视频 | 97超在线视频 | 最近中文字幕高清字幕免费mv | 超碰在线人 | 免费视频一区二区 | 免费男女网站 | 在线精品视频免费播放 | 嫩嫩影院理论片 | 日韩av资源在线观看 | 五月色丁香 | a级一a一级在线观看 | 久久91久久久久麻豆精品 | 亚洲精品女人久久久 | 日韩理论电影在线观看 | 国产精品一区二区久久久久 |