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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

EV3 直接命令 - 第一课 无为的艺术

發布時間:2024/4/11 编程问答 48 豆豆
生活随笔 收集整理的這篇文章主要介紹了 EV3 直接命令 - 第一课 无为的艺术 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

LEGO 的 EV3 是一個極好的游戲工具。它的標準編程方式是 LEGO 的圖形化編程工具。你可以編寫程序,把它們傳到你的 EV3 brick 上,然后啟動它們。但還有另外一種與你的 EV3 交互的方式。把它看作一個服務器并給它發送命令,命令將以數據和/或行為來應答。在這種情形下,你的程序所運行的機器是客戶端。這打開了迷人的新視角。如果程序運行在你的智能手機上,你將獲得很好的交互性和便利性。如果你的客戶端是一個 PC 或筆記本電腦,你將獲得舒服的鍵盤和顯示器。另一種新選項是在一個機器人中結合多個 EV3。一個客戶端與多個服務器通信,這允許機器人具有無限多個馬達和傳感器。或者把你的 EV3 當做一臺機器,生產數據的那種。客戶端可以持續地從 EV3 的傳感器接收數據,這也將打開新的機會之門。如果你想進入這個新世界,你不得不使用 EV3 的直接命令,這需要你的一些工作。如果你準備投資它,則繼續閱讀,否則,開心地玩你的 EV3 并等待其它人做出很酷的新應用吧。

在我們開始討論正式的內容之前,我們先瞥一眼 EV3 的通信協議和 LEGO 直接命令的特性。你的 EV3 提供了三種通信類型,藍牙,WiFI 和 USB。藍牙和 USB 無需其它額外的設備即可使用,通過 WiFi 通信需要一個適配器。所有三種都允許你開發應用程序,運行在任何計算機上與你的 EV3 brick 通信。也許你了解 EV3 的前身,NXT 及其通信協議。它提供了大約 20 個系統命令,類似于函數調用。LEGO 改變了它的理念,EV3 以機器碼提供命令語法。一方面,這意味著實現真正算法的自由,但另一方面,它變得更難入門。

我發現,官方文檔是純粹的技術文檔。他們缺乏動機和教育的方面。我希望這份文檔能成為你深入你的 EV3 的適當入口。如果你已經是一個程序員,或試圖成為一個程序員,你將沉浸在位和字節的世界。如果不是,那可能很困難,但唯一的選擇是,保持雙手分開。試一試,在與樂高 EV3 的有效溝通之下,你將學到很多關于機器代碼,即所有計算機的基礎的知識。

要知道的文檔

LEGO 已經發布了很好很詳細的文檔,你可以在這里下載:http://www.lego.com/en-gb/mindstorms/downloads。對于我們的目的,你絕對需要查看文檔,你可以在標題為 Advanced Users - Developer Kits(PC / MAC) 的標題下找到。EV3 Firmware Developer Kit 是 LEGO EV3 直接命令的參考書。我希望你能深入閱讀它。

有一個 C# 的通信庫,它使用直接命令與 LEGO EV3 通信。如果你喜歡使用開箱即用的軟件,并且喜歡 C#,這可能是你的選擇:http://www.monobrick.dk。

我最初的想法是不發布任何源碼。當功能逐步增長時,編程更有趣。Bugs 是游戲的一部分,查找 bugs 很難,但它是每個編程人員的日常生活。簡短的結論,代碼增長到一定的規模和復雜性時,最初的想法變得不現實。我已經在 Github 上發布了代碼,你可以在這里下載: ev3-python3。

第一課 無為的藝術

你做到了,你真的想介入,那很好!這一課是關于非常基本的通信的知識。我們將實現第一個發送-應答循環。通過 WiFi,藍牙或 USB 給你的 EV3 發送信息,你將獲得一個定義良好的應答。不要屏住呼吸,我們不會從一個令人驚奇的應用程序開始。相反,它什么都不做。這聽起來比實際要做的少,但如果你設法做到這一點,向后傾斜并感到快樂,你就在路上。

比特和字節命名的簡短介紹

也許你已經知道了如何編寫二進制和十六進制數字,大小尾端的含義等等。如果你真的可以把值 156 寫為一個小尾端的 4 字節整數表示格式,則你可以跳過這一節。如果不能,你需要閱讀它,因為你真的需要知道它。

讓我們從基礎開始!幾乎所有的現代計算機都將 8 位組為 1 個字節,并且在內存中按字節尋址(你的 EV3 是一臺現代計算機,因而也是這樣的)。在下文中我們使用如下表示法表示二進制數字:0b 1001 1100。

前導的 0b 告訴我們,后面是二進制的數字,每個字節的 8 個數字被分為 4 和 4 的兩個半個字節。它是數字 156 的二進制表示法,你可以把它看作是:156 = 1128 + 064 + 032 + 116 + 18 + 14 + 02 + 01。可以對相同字節進行另外的解釋。它可以被看作 8 個標記的序列,或可以是符號 £ 的 ASCII 碼。解釋依賴于上下文。目前我們專注于數字。

二進制表示法非常長,因此常見的習慣是把半個字節寫為 16 進制數,其中字母 A 到 F 表示數字 10 到 15。十六進制表示法比較緊湊,它與二進制表示法的轉換很容易。這是因為一個十六進制數表示半個字節。十六進制(這里的值是 156)表示是: 0x 9C。你可以把它看作是:156 = 916 + 121。前導的 0x 告訴我們,后面的是十六進制數。因為它的緊湊,我們可以編寫和閱讀更大的數字。作為一個 4 字節整數,值 156 被寫作:0x 00 00 00 9C。

我們將用冒號 “:” 或豎線 “|” 把字節分開。我們使用豎線表示高級分隔,使用冒號表示低級的。我們將把值為 156 的 2 字節整數寫作:0x|00:9C|。現在我們可以在一行中保存值列表。255(一個無符號 1 字節整數),156(2 字節整數)和 65536(4 字節整數)的序列可以寫為:0x|FF|00:9C|00:01:00:00|。

那負數呢?大多數計算機語言區分有符號和無符號整數。如果整數是有符號的,則它們的第一位是負號標志,整數是另一個范圍的。有符號 1 字節整數的范圍是 -128 到 127,有符號 2 字節整數的范圍是 -32,768 到 32,767 等等。負數值的計算方法是最小值(-128,-32,768 等)加上其余非符號標志的值。有符號 1 字節整數的最小值,-128 寫為 0b 1000 0000 或 0x|80|,有符號 2 字節整數值 -1 (-32,768 + 32,767) 為:0b 1111 1111 1111 1111 或 0x|FF:FF|

那什么是 小尾端 呢?OK,我不再保守這個秘密了。小尾端格式反轉字節的位置(你常常使用的,被稱作 大尾端)。2 字節整數值 156 的小尾端格式寫作:0x|9C:00|。

也許這聽起來像個糟糕的玩笑,但非常抱歉,EV3 直接命令讀和寫所有數字都是以小尾端進行的,那不是我的錯。但我可以給你一些安慰。首先,本課程使用數字。其次,存在管理小尾端數的好工具。在 Python 中,你可以使用 struct 模塊,在 Java 中,ByteBuffer 可能是你選擇的對象。

什么都不做的直接命令

第一個例子展示所有可能的直接命令中最簡單的那個。你將向你的 EV3 發送一條消息,并期待它有所回應。讓我們看一下要發送的消息,它由如下 8 個字節組成:

------------------------- \ len \ cnt \ty\ hd \op\ ------------------------- 0x|06:00|2A:00|00|00:00|01| ------------------------- \ 6 \ 42 \Re\ 0,0 \N \ \ \ \ \ \o \ \ \ \ \ \p \ -------------------------

消息本身是以 0x 開頭的那一行。在消息的頂部,你會看到關于消息對應部分的類型的一些注釋。底部顯示關于其值的注釋。消息的 8 個字節由如下部分組成:

  • 消息長度(字節 0,1):開頭的兩個字節不是直接命令本身的組成部分。它們是通信協議的一部分,在 EV3 的情況下可以是 Wifi,藍牙或 USB。長度被編碼為小尾端格式的 2 字節無符號整數,因此 0x|06:00| 表示值 6。

  • 消息計數器(字節 2,3):接下來的兩個字節是這個直接命令的指紋。消息計數器將包含在應答中,并可以用來匹配直接命令和它的應答。這也是一個小尾端格式的 2 字節無符號整數。在我們的例子中把消息計數器設置為 0x|2A:00|,其值為 42。

  • 消息類型(字節 4):它可以是如下的兩個值之一:

    • DIRECT_COMMAND_REPLY = 0x|00|
    • DIRECT_COMMAND_NO_REPLY = 0x|80|
      在我們的例子中我們希望 EV3 應答消息。
  • 頭部(字節 5,6):接下來的兩個字節,第一個操作之前最后的部分,是頭部。它包含了兩個數字,它們定義了直接命令的內存大小(是的,它是復數,我們有兩個內存,一個局部的和一個全局的)。我們將很快回到這個內存大小的具體細節。此時我們很幸運,我們的命令不需要任何內存,因而我們把頭部設置為 0x|00:00|。

  • 操作(從字節 7 開始):在我們的例子中是單個字節,它表示:opNOP = 0x|01|,什么也不做,EV3 的 idle 操作。

給 EV3 發送消息

我們的任務是,發送上面描述的消息給 EV3。如何做到呢?你可以在三種通信協議中選擇一種,藍牙,Wifi 和 USB,且你可以選擇支持至少一種通信協議的任何編程語言。下面我展示 Python 和 Java 的例子。如果沒有你喜愛的語言,將程序翻譯成你最喜愛的計算機語言并發送給我會很棒。它們將被發布在這里。

藍牙

你需要訪問開啟了藍牙的計算機,且你需要開啟你的 EV3 上的藍牙。接下來你需要為兩個設備做配對。這可以從 EV3 或你的計算機發起。EV3 的用戶指南對此過程有詳細描述。如需幫助,你可以在網上找到教程,這里有 LEGO 頁面的鏈接:http://www.lego.com/en-gb/mindstorms/support/。配對過程將向你展示你的 EV3 的 MAC 地址。你需要注意它。此外,你也可以在你的 EV3 的顯示器中,在 Brick Info / ID 下面讀取 MAC 地址,這里展示的是 EV3 MAC 地址的沒有冒號分割的十六進制形式,如 001653602591,其 MAC 地址為 00:16:53:60:25:91。

python

你需要完成如下的步驟:

  • 把代碼復制到名為 EV3_do_nothing_bluetooth.py 的文件中。
  • 把 MAC 地址從 00:16:53:42:2B:99 變為你的 EV3 的值。
  • 打開一個終端并切換到你的程序的目錄。
  • 通過鍵入 python3 EV3_do_nothing_bluetooth.py 運行它。
#!/usr/bin/env python3import socket import structclass EV3():def __init__(self, host: str):self._socket = socket.socket(socket.AF_BLUETOOTH,socket.SOCK_STREAM,socket.BTPROTO_RFCOMM)self._socket.connect((host, 1))def __del__(self):if isinstance(self._socket, socket.socket):self._socket.close()def send_direct_cmd(self, ops: bytes, local_mem: int=0, global_mem: int=0) -> bytes:cmd = b''.join([struct.pack('<h', len(ops) + 5),struct.pack('<h', 42),DIRECT_COMMAND_REPLY,struct.pack('<h', local_mem*1024 + global_mem),ops])self._socket.send(cmd)print_hex('Sent', cmd)reply = self._socket.recv(5 + global_mem)print_hex('Recv', reply)return replydef print_hex(desc: str, data: bytes) -> None:print(desc + ' 0x|' + ':'.join('{:02X}'.format(byte) for byte in data) + '|')DIRECT_COMMAND_REPLY = b'\x00' opNop = b'\x01' my_ev3 = EV3('00:16:53:42:2B:99') ops_nothing = opNop my_ev3.send_direct_cmd(ops_nothing)

socket 的實現依賴于你計算機的操作系統。如果不支持 AF_BLUETOOTH (你將看到一條錯誤消息如 AttributeError: module ‘socket’ has no attribute ‘AF_BLUETOOTH’),你可以使用 pybluez,那意味著你需要導入 bluetooth 而不是 socket。在我的情況下那是說:

  • 用 pip3 安裝 pybluez:sudo pip3 install pybluez
    安裝 pybluez 有兩個前提條件:一是系統中配置的當前默認 Python 是 Python 3,這這個配置可以通過 $ sudo update-alternatives --config python 完成;二是已經安裝了藍牙開發包 libbluetooth-dev,這可以通過執行命令 $ sudo apt-get install libbluetooth-dev 完成,否則在安裝 pybluez 時將報出如下錯誤:
creating build/lib.linux-x86_64-3.5creating build/lib.linux-x86_64-3.5/bluetoothcopying bluetooth/osx.py -> build/lib.linux-x86_64-3.5/bluetoothcopying bluetooth/__init__.py -> build/lib.linux-x86_64-3.5/bluetoothcopying bluetooth/btcommon.py -> build/lib.linux-x86_64-3.5/bluetoothcopying bluetooth/msbt.py -> build/lib.linux-x86_64-3.5/bluetoothcopying bluetooth/widcomm.py -> build/lib.linux-x86_64-3.5/bluetoothcopying bluetooth/ble.py -> build/lib.linux-x86_64-3.5/bluetoothcopying bluetooth/bluez.py -> build/lib.linux-x86_64-3.5/bluetoothrunning build_extbuilding 'bluetooth._bluetooth' extensioncreating build/temp.linux-x86_64-3.5creating build/temp.linux-x86_64-3.5/bluezx86_64-linux-gnu-gcc -pthread -DNDEBUG -g -fwrapv -O2 -Wall -Wstrict-prototypes -g -fstack-protector-strong -Wformat -Werror=format-security -Wdate-time -D_FORTIFY_SOURCE=2 -fPIC -I./port3 -I/usr/include/python3.5m -c bluez/btmodule.c -o build/temp.linux-x86_64-3.5/bluez/btmodule.oIn file included from bluez/btmodule.c:20:0:bluez/btmodule.h:5:33: fatal error: bluetooth/bluetooth.h: 沒有那個文件或目錄compilation terminated.error: command 'x86_64-linux-gnu-gcc' failed with exit status 1---------------------------------------- Command "/usr/bin/python -u -c "import setuptools, tokenize;__file__='/tmp/pip-install-mk3f5p_f/pybluez/setup.py';f=getattr(tokenize, 'open', open)(__file__);code=f.read().replace('\r\n', '\n');f.close();exec(compile(code, __file__, 'exec'))" install --record /tmp/pip-record-gpk9_xhc/install-record.txt --single-version-externally-managed --compile" failed with error code 1 in /tmp/pip-install-mk3f5p_f/pybluez/
  • 修改程序:
#!/usr/bin/env python3import bluetooth import structclass EV3():def __init__(self, host: str):self._socket = bluetooth.BluetoothSocket(bluetooth.RFCOMM)self._socket.connect((host, 1))def __del__(self):if isinstance(self._socket, bluetooth.BluetoothSocket):self._socket.close() ...
  • 運行程序

  • 關于 struct.pack 的說明
    struct 模塊的 pack() 函數的原型如下:

struct.pack(format, v1, v2, ...)

即第一個參數是格式串,后面的是需要打包的數值。

其中的格式串 format 由兩部分組成,分別是表示字節序/大小/對齊方式的字符和格式字符。表示字節序/大小/對齊方式的各字符含義如下表:

字符字節序大小對齊方式
@本地的本地的本地的
=本地的標準的none
<小尾端標準的none
>大尾端標準的none
!網絡(= 大尾端)標準的none

格式字符的含義如下表:

格式C 類型Python 類型標準大小注意
xpad byteno value
ccharbytes of length 11
bsigned charinteger1(1),(3)
Bunsigned charinteger1(3)
?_Boolbool1(1)
hshortinteger2(3)
Hunsigned shortinteger2(3)
iintinteger4(3)
Iunsigned intinteger4(3)
llonginteger4(3)
Lunsigned longinteger4(3)
qlong longinteger8(2), (3)
Qunsigned long longinteger8(2), (3)
nssize_tinteger(4)
Nsize_tinteger(4)
e(7)float2(5)
ffloatfloat4(5)
ddoublefloat8(5)
schar[]bytes
pchar[]bytes
Pvoid *integer(6)

關于 struct 模塊更詳細的說明可以參考其官方文檔。

Java

與藍牙設備通信,我的選擇是 bluecove。下載 Java 包 bluecove-2.1.0.jar(在 Unix 上也可以是 bluecove-gpl-2.1.0.jar)之后,你可以把它們添加到你的 classpath 上。在我的 Unix 機器上,這通過以下命令完成:

export CLASSPATH=$CLASSPATH:./bluecove-2.1.0.jar:./bluecove-gpl-2.1.0.jar

bluecove-2.1.0.jar 的下載地址為 https://mvnrepository.com/artifact/net.sf.bluecove/bluecove/2.1.0,bluecove-gpl-2.1.0.jar 的下載地址為 https://mvnrepository.com/artifact/net.sf.bluecove/bluecove-gpl/2.1.0。

然后,執行下面的步驟:

  • 把下面的代碼拷貝到名為 EV3_do_nothing_bluetooth.java 的文件中。
  • 把 MAC 地址由 001653422B99 修改為你的 EV3 的值。
  • 打開一個終端并切換到你的程序的目錄。
  • 鍵入 javac EV3_do_nothing_bluetooth.java 編譯它。
  • 鍵入 java EV3_do_nothing_bluetooth 運行它。
import java.io.*; import javax.microedition.io.*;import java.nio.ByteBuffer; import java.nio.ByteOrder;import java.io.*;public class EV3_do_nothing_bluetooth {static final String mac_addr = "001653422B99";static final byte opNop = (byte) 0x01;static final byte DIRECT_COMMAND_REPLY = (byte) 0x00;static InputStream in;static OutputStream out;public static void connectBluetooth ()throws IOException {String s = "btspp://" + mac_addr + ":1";StreamConnection c = (StreamConnection) Connector.open(s);in = c.openInputStream();out = c.openOutputStream();}public static ByteBuffer sendDirectCmd (ByteBuffer operations,int local_mem, int global_mem)throws IOException {ByteBuffer buffer = ByteBuffer.allocateDirect(operations.position() + 7);buffer.order(ByteOrder.LITTLE_ENDIAN);buffer.putShort((short) (operations.position() + 5)); // lengthbuffer.putShort((short) 42); // counterbuffer.put(DIRECT_COMMAND_REPLY); // typebuffer.putShort((short) (local_mem*1024 + global_mem)); // headerfor (int i=0; i < operations.position(); i++) { // operationsbuffer.put(operations.get(i));}byte[] cmd = new byte [buffer.position()];for (int i=0; i<buffer.position(); i++) cmd[i] = buffer.get(i);out.write(cmd);printHex("Sent", buffer);byte[] reply = new byte[global_mem + 5];in.read(reply);buffer = ByteBuffer.wrap(reply);buffer.position(reply.length);printHex("Recv", buffer);return buffer;}public static void printHex(String desc, ByteBuffer buffer) {System.out.print(desc + " 0x|");for (int i= 0; i < buffer.position() - 1; i++) {System.out.printf("%02X:", buffer.get(i));}System.out.printf("%02X|", buffer.get(buffer.position() - 1));System.out.println();}public static void main (String args[] ) {try {connectBluetooth();ByteBuffer operations = ByteBuffer.allocateDirect(1);operations.put(opNop);ByteBuffer reply = sendDirectCmd(operations, 0, 0);}catch (Exception e) {e.printStackTrace(System.err);}} }

這就是一個普通的 Java 應用,可以采用自己順手的構建工具和 IDE。引入 bluecove 和 bluecove-gpl 的 Maven 依賴如下:

<dependencies><dependency><groupId>net.sf.bluecove</groupId><artifactId>bluecove</artifactId><version>2.1.0</version></dependency><dependency><groupId>net.sf.bluecove</groupId><artifactId>bluecove-gpl</artifactId><version>2.1.0</version></dependency></dependencies>

Wifi

你需要一個 Wifi 適配器將你的 EV3 連接到你的本地網絡。下面文檔的第一部分描述了這個過程:http://www.monobrick.dk/guides/how-to-establish-a-wifi-connection-with-the-ev3-brick/。現在你的 EV3 是本地網絡的一部分,且具有一個網絡地址了。從網絡中的所有機器上你都可以與之通信。如上面提到的文檔所描述的那樣,需要如下步驟與 EV3 建立 TCP/IP 連接:

  • 在端口 3015 上監聽來自于 EV3 的 UDP 廣播。
  • 向 EV3 發回一個 UDP 消息使它接受一個 TCP/IP 連接。
  • 在端口 5555 上建立一個 TCP/IP 連接。
  • 通過 TCP/IP 給 EV3 發送一條解鎖消息。

Python

你需要完成如下步驟:

  • 把代碼復制到名為 EV3_do_nothing_wifi.py 的文件中。
  • 打開一個終端,并切換到你的程序的目錄。
  • 鍵入 python3 EV3_do_nothing_wifi.py 運行它。
#!/usr/bin/env python3import socket import struct import reclass EV3():def __init__(self, host: str):# listen on port 3015 for a UDP broadcast from the EV3UDPSock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)UDPSock.bind(('', 3015))data, addr = UDPSock.recvfrom(67)# pick serial number, port, name and protocol# from the broadcast messagematcher = re.search('Serial-Number: (\w*)\s\n' +'Port: (\d{4,4})\s\n' +'Name: (\w+)\s\n' +'Protocol: (\w+)\s\n',data.decode('utf-8'))serial_number = matcher.group(1)port = matcher.group(2)name = matcher.group(3)protocol = matcher.group(4)if serial_number.upper() != host.replace(':', '').upper():self._socket = Noneraise ValueError('found ev3 but not ' + host)# Send an UDP message back to the EV3# to make it accept a TCP/IP connectionUDPSock.sendto(' '.encode('utf-8'), (addr[0], int(port)))UDPSock.close()# Establish a TCP/IP connection with EV3s address and portself._socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)self._socket.connect((addr[0], int(port)))# Send an unlock message to the EV3 over TCP/IPmsg = ''.join(['GET /target?sn=' + serial_number + 'VMTP1.0\n','Protocol: ' + protocol])self._socket.send(msg.encode('utf-8'))reply = self._socket.recv(16).decode('utf-8')if not reply.startswith('Accept:EV340'):raise IOError('No wifi connection to ' + name + ' established')def __del__(self):if isinstance(self._socket, socket.socket):self._socket.close()def send_direct_cmd(self, ops: bytes, local_mem: int=0, global_mem: int=0) -> bytes:cmd = b''.join([struct.pack('<h', len(ops) + 5),struct.pack('<h', 42),DIRECT_COMMAND_REPLY,struct.pack('<h', local_mem*1024 + global_mem),ops])self._socket.send(cmd)print_hex('Sent', cmd)reply = self._socket.recv(5 + global_mem)print_hex('Recv', reply)return replydef print_hex(desc: str, data: bytes) -> None:print(desc + ' 0x|' + ':'.join('{:02X}'.format(byte) for byte in data) + '|')DIRECT_COMMAND_REPLY = b'\x00' opNop = b'\x01' my_ev3 = EV3('00:16:53:42:2B:99') ops_nothing = opNop my_ev3.send_direct_cmd(ops_nothing)

Java

你需要完成如下的步驟:

  • 把代碼復制到名為 EV3_do_nothing_wifi.java 的文件中。
  • 打開一個終端,并切換到你的程序的目錄。
  • 鍵入 javac EV3_do_nothing_wifi.java 編譯它。
  • 鍵入 java EV3_do_nothing_wifi 運行它。
import java.net.Socket; import java.net.SocketException; import java.net.ServerSocket; import java.net.DatagramSocket; import java.net.DatagramPacket; import java.net.InetAddress;import java.nio.ByteBuffer; import java.nio.IntBuffer; import java.nio.ByteOrder;import java.io.*; import java.util.regex.*;public class EV3_do_nothing_wifi {static final byte opNop = (byte) 0x01;static final byte DIRECT_COMMAND_REPLY = (byte) 0x00;static InputStream in;static OutputStream out;public static void connectWifi ()throws IOException, SocketException {// listen for a UDP broadcast from the EV3 on port 3015DatagramSocket listener = new DatagramSocket(3015);DatagramPacket packet_r = new DatagramPacket(new byte[67], 67);listener.receive(packet_r); // receive the broadcast messageString broadcast_message = new String(packet_r.getData());/* pick serial number, port, name and protocol from the broadcast message */Pattern broadcast_pattern = Pattern.compile("Serial-Number: (\\w*)\\s\\n" +"Port:\\s(\\d{4,4})\\s\\n" +"Name:\\s(\\w+)\\s\\n" +"Protocol:\\s(\\w+)\\s\\n");Matcher matcher = broadcast_pattern.matcher(broadcast_message);String serial_number, name, protocol;int port;if(matcher.matches()) {serial_number = matcher.group(1);port = Integer.valueOf(matcher.group(2));name = matcher.group(3);protocol = matcher.group(4);}else {throw new IOException("Unexpected Broadcast message: " + broadcast_message);}InetAddress adr = packet_r.getAddress();// connect the EV3 with its address and portlistener.connect(adr, port);/* Send an UDP message back to the EV3 to make it accept a TCP/IP connection */listener.send(new DatagramPacket(new byte[1], 1));// close the UDP connectionlistener.close();// Establish a TCP/IP connection with EV3s address and portSocket socket = new Socket(adr, port);in = socket.getInputStream();out = socket.getOutputStream();// Send an unlock message to the EV3 over TCP/IPString unlock_message = "GET /target?sn=" + serial_number + "VMTP1.0\n" + "Protocol: " + protocol;out.write(unlock_message.getBytes());byte[] reply = new byte[16]; // read replyin.read(reply);if (! (new String(reply)).startsWith("Accept:EV340")) {throw new IOException("No wifi connection established " + name);} }public static ByteBuffer sendDirectCmd (ByteBuffer operations,int local_mem, int global_mem)throws IOException {ByteBuffer buffer = ByteBuffer.allocateDirect(operations.position() + 7);buffer.order(ByteOrder.LITTLE_ENDIAN);buffer.putShort((short) (operations.position() + 5)); // lengthbuffer.putShort((short) 42); // counterbuffer.put(DIRECT_COMMAND_REPLY); // typebuffer.putShort((short) (local_mem*1024 + global_mem)); // headerfor (int i=0; i<operations.position(); i++) { // operationsbuffer.put(operations.get(i));}byte[] cmd = new byte [buffer.position()];for (int i=0; i<buffer.position(); i++) cmd[i] = buffer.get(i);out.write(cmd);printHex("Sent", buffer);byte[] reply = new byte[global_mem + 5];in.read(reply);buffer = ByteBuffer.wrap(reply);buffer.position(reply.length);printHex("Recv", buffer);return buffer;}public static void printHex(String desc, ByteBuffer buffer) {System.out.print(desc + " 0x|");for (int i= 0; i<buffer.position() - 1; i++) {System.out.printf("%02X:", buffer.get(i));}System.out.printf("%02X|", buffer.get(buffer.position() - 1));System.out.println();}public static void main (String args[] ) {try {connectWifi();ByteBuffer operations = ByteBuffer.allocateDirect(1);operations.put(opNop);ByteBuffer reply = sendDirectCmd(operations, 0, 0);}catch (Exception e) {e.printStackTrace(System.err);}} }

USB

通用串行總線是一個連接電子設備的工業標準。你的 EV3 具有一個 2.0 的Mini-B 接口(標有 PC 字樣)。這是性能最好的通信協議,但它需要一條線。在你運行你的程序的電腦上,你需要有與 LEGO EV3 通信的權限。在 Ubuntu Linux 上,通過 lsusb 查看當前已經連接的 USB 設備,以確認 EV3 被成功識別,如:

$ lsusb Bus 002 Device 002: ID 8087:8000 Intel Corp. Bus 002 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub Bus 001 Device 002: ID 8087:8008 Intel Corp. Bus 001 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub Bus 004 Device 001: ID 1d6b:0003 Linux Foundation 3.0 root hub Bus 003 Device 006: ID 5986:055c Acer, Inc Bus 003 Device 005: ID 8087:07dc Intel Corp. Bus 003 Device 004: ID 2717:ff48 Bus 003 Device 003: ID 046d:c52b Logitech, Inc. Unifying Receiver Bus 003 Device 007: ID 0694:0005 Lego Group Bus 003 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub

由 Bus 003 Device 007: ID 0694:0005 Lego Group 這一行可以看出 EV3 已經被系統成功識別了,其中的 Bus 003 Device 007 部分描述了設備連接的位置,ID 0694:0005 描述生產商 ID 和產品 ID。默認情況下,新連接的新類型的 USB 設備只有 root 用戶有訪問權限,這一點可以根據 Bus 003 Device 007 這些信息,查看 /dev/bus/usb 下對應的設備文件看出來:

$ ls -al /dev/bus/usb/003/ 總用量 0 drwxr-xr-x 2 root root 160 11月 1 09:51 . drwxr-xr-x 6 root root 120 11月 1 09:49 .. crw-rw-r-- 1 root root 189, 256 11月 1 09:49 001 crw-rw-r-- 1 root root 189, 258 11月 1 09:49 003 crw-rw----+ 1 root audio 189, 259 11月 1 09:50 004 crw-rw-r-- 1 root root 189, 260 11月 1 09:49 005 crw-rw-r-- 1 root root 189, 261 11月 1 09:49 006 crw-rw-r-- 1 root root 189, 262 11月 1 09:51 007

可以看到對應于 LEGO EV3 的 USB 設備文件 /dev/bus/usb/003/007,其所有者和所有者組都為 root,且其它用戶只有讀權限。當我們在沒有權限的情況下,通過 pyusb 訪問 EV3 時,將報出如下的錯誤:

Traceback (most recent call last):File "/home/hanpfei0306/data/MyProjects/wolfcs-tools/EV3_do_nothing_usb.py", line 44, in <module>my_ev3 = EV3('00:16:53:42:2B:99')File "/home/hanpfei0306/data/MyProjects/wolfcs-tools/EV3_do_nothing_usb.py", line 11, in __init__serial_number = usb.util.get_string(self._device, self._device.iSerialNumber)File "/usr/local/lib/python3.5/dist-packages/usb/util.py", line 314, in get_stringraise ValueError("The device has no langid") ValueError: The device has no langid

為了使得普通的非 root 用戶也能通過 USB 訪問 EV3,需要為它添加 udev 規則。在我的 Ubuntu Linux 16.04 上是,創建文件 /etc/udev/rules.d/51-legoev3-usb.rules,并添加如下內容:

SUBSYSTEM=="usb", ATTR{idVendor}=="0694", ATTR{idProduct}=="0005", MODE="0666", GROUP="<group>"

把 <group> 修改為你所屬的那個。添加了 udev 規則之后,拔出 USB 接口重新插入即可使規則生效。此時即使在普通用戶下,通過 lsusb 也能查看到關于 EV3 的詳細信息:

$ lsusb -v -d 0694:0005 Bus 003 Device 013: ID 0694:0005 Lego Group Device Descriptor:bLength 18bDescriptorType 1bcdUSB 2.00bDeviceClass 0 (Defined at Interface level)bDeviceSubClass 0 bDeviceProtocol 0 bMaxPacketSize0 64idVendor 0x0694 Lego GroupidProduct 0x0005 bcdDevice 2.16iManufacturer 1 LEGO GroupiProduct 2 EV3iSerial 3 001653602591bNumConfigurations 1Configuration Descriptor:bLength 9bDescriptorType 2wTotalLength 41bNumInterfaces 1bConfigurationValue 1iConfiguration 1 LEGO GroupbmAttributes 0xc0Self PoweredMaxPower 2mAInterface Descriptor:bLength 9bDescriptorType 4bInterfaceNumber 0bAlternateSetting 0bNumEndpoints 2bInterfaceClass 3 Human Interface DevicebInterfaceSubClass 0 No SubclassbInterfaceProtocol 0 NoneiInterface 4 Xfer data to and from EV3 brickHID Device Descriptor:bLength 9bDescriptorType 33bcdHID 1.10bCountryCode 0 Not supportedbNumDescriptors 1bDescriptorType 34 ReportwDescriptorLength 29Report Descriptor: (length is 29)Item(Global): Usage Page, data= [ 0x00 0xff ] 65280(null)Item(Local ): Usage, data= [ 0x01 ] 1(null)Item(Main ): Collection, data= [ 0x01 ] 1ApplicationItem(Global): Logical Minimum, data= [ 0x00 ] 0Item(Global): Logical Maximum, data= [ 0xff 0x00 ] 255Item(Global): Report Size, data= [ 0x08 ] 8Item(Global): Report Count, data= [ 0x00 0x04 ] 1024Item(Local ): Usage, data= [ 0x01 ] 1(null)Item(Main ): Input, data= [ 0x02 ] 2Data Variable Absolute No_Wrap LinearPreferred_State No_Null_Position Non_Volatile BitfieldItem(Global): Report Count, data= [ 0x00 0x04 ] 1024Item(Local ): Usage, data= [ 0x01 ] 1(null)Item(Main ): Output, data= [ 0x02 ] 2Data Variable Absolute No_Wrap LinearPreferred_State No_Null_Position Non_Volatile BitfieldItem(Main ): End Collection, data=noneEndpoint Descriptor:bLength 7bDescriptorType 5bEndpointAddress 0x81 EP 1 INbmAttributes 3Transfer Type InterruptSynch Type NoneUsage Type DatawMaxPacketSize 0x0400 1x 1024 bytesbInterval 4Endpoint Descriptor:bLength 7bDescriptorType 5bEndpointAddress 0x01 EP 1 OUTbmAttributes 3Transfer Type InterruptSynch Type NoneUsage Type DatawMaxPacketSize 0x0400 1x 1024 bytesbInterval 4 Device Qualifier (for other device speed):bLength 10bDescriptorType 6bcdUSB 2.00bDeviceClass 0 (Defined at Interface level)bDeviceSubClass 0 bDeviceProtocol 0 bMaxPacketSize0 64bNumConfigurations 1 Device Status: 0x0001Self Powered

更多信息可參考如何在非root用戶下,訪問普通的usb設備。

通過它們的 vendor-id 0x0694 和 product-id 0x0005 標識 EV3 設備。模式 0666 允許屬于 <group> 的所有用戶具有讀和寫權限。EV3-USB 設備描述符顯示了一個具有一個接口和兩個端點的配置,0x01 用于給 EV3 發送數據,0x81 用于從 EV3 接收數據。數據以 1024 字節的包發送和接收。

Python

也許你需要安裝 pyusb。對于我的系統來說,這通過如下命令完成:

sudo pip3 install --pre pyusb

如果已經安裝了 pyusb,你需要完成如下的步驟:

  • 把代碼復制到名為 EV3_do_nothing_usb.py 的文件中。
  • 打開一個終端,并切換到你的程序的目錄。
  • 鍵入 python3 EV3_do_nothing_usb.py 運行它。
#!/usr/bin/env python3import usb.core import structclass EV3():def __init__(self, host: str):self._device = usb.core.find(idVendor=ID_VENDOR_LEGO, idProduct=ID_PRODUCT_EV3)if self._device is None:raise RuntimeError("No Lego EV3 found")serial_number = usb.util.get_string(self._device, self._device.iSerialNumber)if serial_number.upper() != host.replace(':', '').upper():raise ValueError('found ev3 but not ' + host)if self._device.is_kernel_driver_active(0) is True:self._device.detach_kernel_driver(0)self._device.set_configuration()self._device.read(EP_IN, 1024, 100)def __del__(self): passdef send_direct_cmd(self, ops: bytes, local_mem: int=0, global_mem: int=0) -> bytes:cmd = b''.join([struct.pack('<h', len(ops) + 5),struct.pack('<h', 42),DIRECT_COMMAND_REPLY,struct.pack('<h', local_mem*1024 + global_mem),ops])self._device.write(EP_OUT, cmd, 100)print_hex('Sent', cmd)reply = self._device.read(EP_IN, 1024, 100)[0:5+global_mem]print_hex('Recv', reply)return replydef print_hex(desc: str, data: bytes) -> None:print(desc + ' 0x|' + ':'.join('{:02X}'.format(byte) for byte in data) + '|')ID_VENDOR_LEGO = 0x0694 ID_PRODUCT_EV3 = 0x0005 EP_IN = 0x81 EP_OUT = 0x01 DIRECT_COMMAND_REPLY = b'\x00' opNop = b'\x01' my_ev3 = EV3('00:16:53:42:2B:99') ops_nothing = opNop my_ev3.send_direct_cmd(ops_nothing)

譯者注:

  • 由上面通過 lsusb 扒出來的 EV3 設備信息可以看到,EV3 的串號是 iSerial 3 001653602591,即 00:16:53:60:25:91,而不是 00:16:53:42:2B:99',因此上面代碼中串號判斷的幾行:
serial_number = usb.util.get_string(self._device, self._device.iSerialNumber)if serial_number.upper() != host.replace(':', '').upper():raise ValueError('found ev3 but not ' + host)

可以去掉。

  • 在 LEGO EV3 設備初次連接之后,執行上述代碼,可以正常的向 EV3 發送消息并接收響應。但再次執行時,程序報出如下的錯誤:
Traceback (most recent call last):File "/home/hanpfei0306/data/MyProjects/wolfcs-tools/EV3_do_nothing_usb.py", line 44, in <module>my_ev3 = EV3('00:16:53:42:2B:99')File "/home/hanpfei0306/data/MyProjects/wolfcs-tools/EV3_do_nothing_usb.py", line 17, in __init__self._device.read(EP_IN, 1024, 100)File "/usr/local/lib/python3.5/dist-packages/usb/core.py", line 988, in readself.__get_timeout(timeout))File "/usr/local/lib/python3.5/dist-packages/usb/backend/libusb1.py", line 851, in intr_readtimeout)File "/usr/local/lib/python3.5/dist-packages/usb/backend/libusb1.py", line 936, in __read_check(retval)File "/usr/local/lib/python3.5/dist-packages/usb/backend/libusb1.py", line 595, in _checkraise USBError(_strerror(ret), ret, _libusb_errno[ret]) usb.core.USBError: [Errno 110] Operation timed out

StackOverflow 上對這個問題有所討論:https://stackoverflow.com/questions/38658907/trouble-using-pyusb-to-read-write-from-usb-device-timeouts。這個問題可以通過在 find() 方法調用之后,加入 self._device.reset() 來解決,如:

def __init__(self, host: str):self._device = usb.core.find(idVendor=ID_VENDOR_LEGO, idProduct=ID_PRODUCT_EV3)if self._device is None:raise RuntimeError("No Lego EV3 found")self._device.reset()serial_number = usb.util.get_string(self._device, self._device.iSerialNumber)# if serial_number.upper() != host.replace(':', '').upper():# raise ValueError('found ev3 but not ' + host)if self._device.is_kernel_driver_active(0) is True:self._device.detach_kernel_driver(0)self._device.set_configuration()self._device.read(EP_IN, 1024, 100)

Java

我選擇的與 USB 設備通信的是 usb4java。下載了 Java 包之后,你可以把它們添加到你的 classpath 中。在我的 Unix 機器是,通過如下命令完成:

export CLASSPATH=$CLASSPATH:./usb4java-1.3.0.jar:./libusb4java-1.3.0-linux-x86_64.jar

譯者注:
如果構建系統用的是 Maven,也可以通過在 pom.xml 文件中添加如下的依賴來使用 usb4java:

<dependency><groupId>org.usb4java</groupId><artifactId>usb4java</artifactId><version>1.3.0</version></dependency>

然后,執行下面的步驟:

  • 把下面的代碼拷貝到名為 EV3_do_nothing_usb.java 的文件中。
  • 打開一個終端并切換到你的程序的目錄。
  • 鍵入 javac EV3_do_nothing_usb.java 編譯它。
  • 鍵入 java EV3_do_nothing_usb 運行它。
import org.usb4java.Device; import org.usb4java.DeviceDescriptor; import org.usb4java.DeviceHandle; import org.usb4java.DeviceList; import org.usb4java.LibUsb; import org.usb4java.LibUsbException;import java.nio.ByteBuffer; import java.nio.IntBuffer; import java.nio.ByteOrder;public class EV3_do_nothing_usb {static final short ID_VENDOR_LEGO = (short) 0x0694;static final short ID_PRODUCT_EV3 = (short) 0x0005;static final byte EP_IN = (byte) 0x81;static final byte EP_OUT = (byte) 0x01;static final byte opNop = (byte) 0x01;static final byte DIRECT_COMMAND_REPLY = (byte) 0x00;static DeviceHandle handle;public static void connectUsb () {int result = LibUsb.init(null);Device device = null;DeviceList list = new DeviceList();result = LibUsb.getDeviceList(null, list);if (result < 0){throw new RuntimeException("Unable to get device list. Result=" + result);}boolean found = false;for (Device dev: list) {DeviceDescriptor descriptor = new DeviceDescriptor();result = LibUsb.getDeviceDescriptor(dev, descriptor);if (result != LibUsb.SUCCESS) {throw new LibUsbException("Unable to read device descriptor", result);}if ( descriptor.idVendor() == ID_VENDOR_LEGO|| descriptor.idProduct() == ID_PRODUCT_EV3) {device = dev;found = true;break;}}LibUsb.freeDeviceList(list, true);if (! found) throw new RuntimeException("Lego EV3 device not found.");handle = new DeviceHandle();result = LibUsb.open(device, handle);if (result != LibUsb.SUCCESS) {throw new LibUsbException("Unable to open USB device", result);}boolean detach = LibUsb.kernelDriverActive(handle, 0) != 0;if (detach) result = LibUsb.detachKernelDriver(handle, 0);if (result != LibUsb.SUCCESS) {throw new LibUsbException("Unable to detach kernel driver", result);}result = LibUsb.claimInterface(handle, 0);if (result != LibUsb.SUCCESS) {throw new LibUsbException("Unable to claim interface", result);}}public static ByteBuffer sendDirectCmd (ByteBuffer operations,int local_mem, int global_mem) {ByteBuffer buffer = ByteBuffer.allocateDirect(operations.position() + 7);buffer.order(ByteOrder.LITTLE_ENDIAN);buffer.putShort((short) (operations.position() + 5)); // lengthbuffer.putShort((short) 42); // counterbuffer.put(DIRECT_COMMAND_REPLY); // typebuffer.putShort((short) (local_mem*1024 + global_mem)); // headerfor (int i=0; i < operations.position(); i++) { // operationsbuffer.put(operations.get(i));}IntBuffer transferred = IntBuffer.allocate(1);int result = LibUsb.bulkTransfer(handle, EP_OUT, buffer, transferred, 100); if (result != LibUsb.SUCCESS) {throw new LibUsbException("Unable to write data", transferred.get(0));}printHex("Sent", buffer);buffer = ByteBuffer.allocateDirect(1024);transferred = IntBuffer.allocate(1);result = LibUsb.bulkTransfer(handle, EP_IN, buffer, transferred, 100);if (result != LibUsb.SUCCESS) {throw new LibUsbException("Unable to read data", result);}buffer.position(global_mem + 5);printHex("Recv", buffer);return buffer;}public static void printHex(String desc, ByteBuffer buffer) {System.out.print(desc + " 0x|");for (int i= 0; i < buffer.position() - 1; i++) {System.out.printf("%02X:", buffer.get(i));}System.out.printf("%02X|", buffer.get(buffer.position() - 1));System.out.println();}public static void main (String args[] ) {try {connectUsb();ByteBuffer operations = ByteBuffer.allocateDirect(1);operations.put(opNop);ByteBuffer reply = sendDirectCmd(operations, 0, 0);LibUsb.releaseInterface(handle, 0);LibUsb.close(handle);}catch (Exception e) {e.printStackTrace(System.err);}} }

應答消息

如果你使用上面方案中的一個成功實現了與 EV3 的通信,則會獲得以下輸出,即直接命令的回復消息:

---------------- \ len \ cnt \rs\ –--------------- 0x|03:00|2A:00|02| –--------------- \ 3 \ 42 \ok\ ----------------

前兩個字節是眾所周知的,它是回復消息消息長度的小尾端表示。在我們的情況中,回復消息是 3 字節長的。接下來的兩個字節是消息計數器,也是廣為人知的,即你發送的消息的指紋,是 42。

最后一個字節是返回狀態,其有 2 個可能值是:

  • DIRECT_REPLY = 0x|02|:直接命令操作成功
  • DIRECT_REPLY_ERROR = 0x|04|:直接命令以失敗結束

如果你真的收到了這條回復消息,那么你入門了。恭喜!

頭部的細節

上面我們跳過了頭部細節的描述。提到了它,頭部包含兩個數,它們定義內存的大小。

第一個數是局部內存的大小,它是你可以在其中保存中間信息的地址空間。第二個數描述了全局內存的大小,它是輸出的地址空間。在 DIRECT_COMMAND_REPLY 的情形中,全局內存將作為回復消息的一部分發回。

局部內存具有最大 63 個字節,全局內存具有最大 1019 字節。那意味著,局部內存大小需要 6 位,全局內存大小需要 10 位。如果一個字節共用的話,所有的組合在一起可以包含在兩個字節中。確實這樣做了。如果以相反的順序寫入頭字節,這是熟悉的大尾端,并且以二進制表示法寫為半字節組,則得到:0b LLLL LLGG GGGG GGGG。開頭的 6 位是局部內存大小,其范圍為 0-63。尾部的 10 位是全局內存大小,其范圍為 0-1020。小尾端下是:0b GGGG GGGG LLLL LLGG。比如如果你的全局內存具有 6 個字節,你的局部內存需要 16 字節,則你的頭部是 0b 0000 0110 0100 0000 或以十六進制表示是 0x 06 40。

這是描述性版本,現在是聲明式方式的第二種方法。如果 local_mem 是本地內存大小,global_mem 是全局內存大小,則計算:header = local_mem * 1024 + global_mem。把頭部以小尾端 2 字節整數格式寫入,你將得到兩個頭部字節。如果你還有疑問,請等待接下來的課程,你將看到大量的頭部并從例子中學習,這將有望解答你的疑問。

什么都不做的變體

在離開我們的第一個例子并關閉第一章之前,我們將測試兩個頭部的變體。第一個嘗試是直接命令具有 6 字節的全局內存空間:

\ len \ cnt \ty\ hd \op\ ------------------------- 0x|06:00|2A:00|00|06:00|01| ------------------------- \ 6 \ 42 \Re\ 0,6 \N \ \ \ \ \ \o \ \ \ \ \ \p \ -------------------------

我們期望獲得 6 字節低值輸出的回復。 因此,你必須將答案的長度從 5 增加到 11。如果這樣做,你將得到:

---------------------------------- \ len \ cnt \rs\ Output \ –--------------------------------- 0x|09:00|2A:00|02|00:00:00:00:00:00| –--------------------------------- \ 9 \ 42 \ok\ \ ----------------------------------

我們添加 16 字節的局部內存空間,并將直接命令更改為以下內容:

------------------------- \ len \ cnt \ty\ hd \op\ ------------------------- 0x|06:00|2A:00|00|06:40|01| ------------------------- \ 6 \ 42 \Re\16,6 \N \ \ \ \ \ \o \ \ \ \ \ \p \ -------------------------

我們希望得到與上述相同的回復,實際上是:

---------------------------------- \ len \ cnt \rs\ Output \ –--------------------------------- 0x|09:00|2A:00|02|00:00:00:00:00:00| –--------------------------------- \ 9 \ 42 \ok\ \ ----------------------------------

你的家庭作業

在進行第 2 課之前,你應該完成如下的家庭作業:

  • 把一個小程序轉換為你喜歡的編程語言并把它集成進你喜歡的開發環境。
  • 準備一些工具,因為一遍又一遍的從頭開始可不是一件讓人愉快的事情。我想到了以下設計:
    • EV3 是一個類。
    • BLUETOOTH,USB,WIFI,STD,ASYNC,SYNC 和 opNop 是公共常量。
    • 連接 EV3 是 EV3 對象初始化的一部分,即協議的選擇通過以特定的參數調用 EV3 對象的構造函數完成。EV3 對象需要記住它的協議類型。socket 和 device 是 EV3 對象的 private 或 protected 變量。
    • 給 EV3 發送數據通過 EV3 類的方法 send_direct_cmd 完成。你可以把示例的函數當作藍圖,但在內部你一定要區分協議。
    • 為了從 EV3 接收數據,我們使用方法 wait_for_reply。你必須把函數 send_direct_cmd 的代碼拆分為兩個新的方法 send_direct_cmd 和 wait_for_reply。
    • 添加一個屬性 verbosity,它控制是否打印已發送的直接命令和收到的回復。
    • 添加一個屬性 sync_mode,它通過如下值控制通信的行為:
      • SYNC:總是使用類型 DIRECT_COMMAND_REPLY 并等待響應。
      • ASYNC:從不等待響應,當不使用全局內存時設置 DIRECT_COMMAND_NO_REPLY,其它情況設置 DIRECT_COMMAND_REPLY。
      • STD:像 ASYNC 那樣設置 DIRECT_COMMAND_NO_REPLY 或 DIRECT_COMMAND_REPLY,但在 DIRECT_COMMAND_REPLY 的情況下等待響應。
    • msg_cnt 是 EV3 對象的私有變量,每次調用 send_direct_cmd 這個值都會增加。使用它來設置消息計數器。
    • 如例子中那樣,消息長度,消息計數器,消息類型和頭部都是在 send_direct_cmd 內部自動添加的。因此 send_direct_cmd 方法的參數 ops 真正地保存操作有關的信息而沒有其它東西。
  • 做一些性能測試,并比較三種通信協議(你將看到,USB 最快,藍牙最慢,Wifi 居于中間,但你可能會賭三個協議之間的絕對值和因素)。
    • 以 DIRECT_COMMAND_REPLY 重復發送 opNop 并計算一個發送接收循環的平均時間。
    • 把連接的時間從發送和接收的時間中分離出來。你將只連接一次,但發送和接收循環的性能將限制你的應用程序。

譯者注:
以我本人手邊的設備,比較方便測試 USB 和藍牙的性能。我測試得到的,noop 操作的連接建立和消息發送性能如下:

連接方式連接時間(單位/秒)消息發送接收時間(單位/秒)
藍牙3.8490.0872
USB0.32040.004

結論

你開始編寫一個類 EV3,用于使用直接命令與 LEGO EV3 通信。這個類允許自由地選擇通信協議,并提供藍牙,USB 和 Wifi。我選擇的編程語言是 Python3。我使用 pydoc3 來展示我們的工程的實際狀態。我希望,你可以簡單地把它轉為你喜歡的語言。此刻,我們的類 EV3 具有如下的 API:

Help on module ev3:NAMEev3 - LEGO EV3 direct commandsCLASSESbuiltins.objectEV3class EV3(builtins.object)| object to communicate with a LEGO EV3 using direct commands| | Methods defined here:| | __del__(self)| closes the connection to the LEGO EV3| | __init__(self, protocol:str, host:str)| Establish a connection to a LEGO EV3 device| | Arguments:| protocol: 'Bluetooth', 'Usb' or 'Wifi'| host: mac-address of the LEGO EV3 (f.i. '00:16:53:42:2B:99')| | send_direct_cmd(self, ops:bytes, local_mem:int=0, global_mem:int=0) -> bytes| Send a direct command to the LEGO EV3| | Arguments:| ops: holds netto data only (operations), the following fields are added:| length: 2 bytes, little endian| counter: 2 bytes, little endian| type: 1 byte, DIRECT_COMMAND_REPLY or DIRECT_COMMAND_NO_REPLY| header: 2 bytes, holds sizes of local and global memory| | Keyword Arguments:| local_mem: size of the local memory| global_mem: size of the global memory| | Returns: | sync_mode is STD: reply (if global_mem > 0) or message counter| sync_mode is ASYNC: message counter| sync_mode is SYNC: reply of the LEGO EV3| | wait_for_reply(self, counter:bytes) -> bytes| Ask the LEGO EV3 for a reply and wait until it is received| | Arguments:| counter: is the message counter of the corresponding send_direct_cmd| | Returns:| reply to the direct command| | ----------------------------------------------------------------------| Data descriptors defined here:| | __dict__| dictionary for instance variables (if defined)| | __weakref__| list of weak references to the object (if defined)| | sync_mode| sync mode (standard, asynchronous, synchronous)| | STD: Use DIRECT_COMMAND_REPLY if global_mem > 0,| wait for reply if there is one.| ASYNC: Use DIRECT_COMMAND_REPLY if global_mem > 0,| never wait for reply (it's the task of the calling program).| SYNC: Always use DIRECT_COMMAND_REPLY and wait for reply.| | The general idea is:| ASYNC: Interruption or EV3 device queues direct commands,| control directly comes back.| SYNC: EV3 device is blocked until direct command is finished,| control comes back, when direct command is finished. | STD: NO_REPLY like ASYNC with interruption or EV3 queuing,| REPLY like SYNC, synchronicity of program and EV3 device.| | verbosity| level of verbosity (prints on stdout).DATABLUETOOTH = 'Bluetooth'USB = 'Usb'WIFI = 'Wifi'STD = 'STD'ASYNC = 'ASYSNC'SYNC = 'SYNC'opNop = b'\x01'

我的 EV3 類是模塊 ev3 的一部分,文件名是 ev3.py。我使用如下程序測試我的 EV3 類:

#!/usr/bin/env python3import ev3my_ev3 = ev3.EV3(protocol=ev3.BLUETOOTH, host='00:16:53:42:2B:99') my_ev3.verbosity = 1ops = ev3.opNopprint("*** SYNC ***") my_ev3.sync_mode = ev3.SYNC my_ev3.send_direct_cmd(ops)print("*** ASYNC (no reply) ***") my_ev3.sync_mode = ev3.ASYNC my_ev3.send_direct_cmd(ops)print("*** ASYNC (reply) ***") counter_1 = my_ev3.send_direct_cmd(ops, global_mem=1) counter_2 = my_ev3.send_direct_cmd(ops, global_mem=1) my_ev3.wait_for_reply(counter_1) my_ev3.wait_for_reply(counter_2)print("*** STD (no reply) ***") my_ev3.sync_mode = ev3.STD my_ev3.send_direct_cmd(ops)print("*** STD (reply) ***") my_ev3.send_direct_cmd(ops, global_mem=5) my_ev3.send_direct_cmd(ops, global_mem=5)

得到輸出:

*** SYNC *** 15:15:05.084275 Sent 0x|06:00|2A:00|00|00:00|01| 15:15:05.168023 Recv 0x|03:00|2A:00|02| *** ASYNC (no reply) *** 15:15:05.168548 Sent 0x|06:00|2B:00|80|00:00|01| *** ASYNC (reply) *** 15:15:05.168976 Sent 0x|06:00|2C:00|00|01:00|01| 15:15:05.169315 Sent 0x|06:00|2D:00|00|01:00|01| 15:15:05.212077 Recv 0x|04:00|2C:00|02|00| 15:15:05.212708 Recv 0x|04:00|2D:00|02|00| *** STD (no reply) *** 15:15:05.213034 Sent 0x|06:00|2E:00|80|00:00|01| *** STD (reply) *** 15:15:05.213411 Sent 0x|06:00|2F:00|00|05:00|01| 15:15:05.254032 Recv 0x|08:00|2F:00|02|00:00:00:00:00| 15:15:05.254633 Sent 0x|06:00|30:00|00|05:00|01| 15:15:05.313027 Recv 0x|08:00|30:00|02|00:00:00:00:00|

一些備注:

  • sync_mode = SYNC 設置 type = DIRECT_COMMAND_REPLY 并自動地等待響應,ok。
  • sync_mode = ASYNC 設置 type = DIRECT_COMMAND_NO_REPLY,不等待,ok。
  • 全局內存設置 type = DIRECT_COMMAND_REPLY 時 sync_mode = ASYNC,不等待。顯式地調用 wait_for_reply 方法獲得響應。
  • 請特別尊重此變體。我們發送兩個直接命令,它們都被執行,且 EV3 設備保存響應。
  • 當我們稍后詢問響應時,我們首先讀取第一條命令的響應。它似乎就像 EV3 是一個 FIFO(先進先出)。
  • 但它也可以并行執行,但它也可以按照完成執行的順序執行并行和重復。我們將回到這一點。
  • 模式 ASYNC 需要一些規律。如果你忘記了讀取響應,當你等待另一件事時,它會來。 我們使用消息計數器來揭示這種情況!
  • 請小心 USB 協議!協議USB請小心! 如果像我一樣直接發送異步命令,這可能會太快。
  • sync_mode = STD,沒有全局內存設置 type = DIRECT_COMMAND_NO_REPLY,并且不等待回復,ok。
  • sync_mode = STD,全局內存設置 type = DIRECT_COMMAND_REPLY,每個直接命令等待響應,ok。
  • 消息計數器隨直接命令遞增,ok。
  • 頭部正確保存全局內存的大小,ok。

如果你完成了作業,那么你已經為新的冒險做好了充分的準備。 我希望在下一課中再次見到你。

原文

總結

以上是生活随笔為你收集整理的EV3 直接命令 - 第一课 无为的艺术的全部內容,希望文章能夠幫你解決所遇到的問題。

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

天天干,天天射,天天操,天天摸 | 黄色在线小网站 | 亚州精品一二三区 | 狠狠黄 | 国产精品videossex国产高清 | 国产看片免费 | 国产小视频在线免费观看 | 久久久久国产一区二区 | 丁香久久激情 | 欧美激情视频一区二区三区免费 | 91九色在线 | 亚洲狠狠婷婷综合久久久 | 婷婷色亚洲| 在线韩国电影免费观影完整版 | 人人澡人人添人人爽一区二区 | 久久婷婷一区 | 亚洲黄色片一级 | 黄色三级在线观看 | 久久久久久久免费 | 亚洲国产精品久久久久久 | 亚洲精品视频免费在线 | 久久精品一区二区三区视频 | www狠狠操 | 2019中文字幕网站 | 最新中文字幕 | 日韩99热 | 午夜男人影院 | 成人午夜网 | 国产精品国产三级国产不产一地 | 一色屋精品视频在线观看 | 在线观看视频黄 | 一区三区视频 | 久久精品爱爱视频 | 免费看污片 | 99色免费 | 欧美精品一区二区免费 | 91人人视频在线观看 | 一区二区三区视频网站 | 91手机电视| 久久人人爽人人片av | 99一级片 | 久久欧美综合 | 亚洲人成网站精品片在线观看 | 欧美激情精品久久 | 久久久精品国产一区二区电影四季 | 亚洲视频在线观看 | 国产午夜av | 超碰97在线人人 | 最近中文字幕视频完整版 | 人人爱天天操 | 欧美午夜视频在线 | 人交video另类hd| 香蕉视频导航 | 国产99免费 | 国产黄色在线看 | 日韩欧美一区二区三区视频 | 91色吧 | 成人小电影在线看 | 国产五码一区 | 成人影片在线免费观看 | 免费看黄在线看 | 日本精品久久久一区二区三区 | 天堂视频中文在线 | 中国美女一级看片 | 激情视频免费在线观看 | 深爱激情久久 | 亚洲黄色在线播放 | 97国产一区二区 | 国产精品视频999 | 九色一区二区 | 国产日韩精品一区二区三区 | 欧美做受xxx | 天天草天天 | 国产一区麻豆 | 国产一区二区观看 | 中文字幕高清在线播放 | 婷婷激情综合五月天 | 午夜精品久久久久久久久久 | 在线观看深夜视频 | 西西人体www444| 色久网| 日韩在线高清免费视频 | 久久毛片高清国产 | 美女免费网站 | 国产精品手机在线观看 | 久久91久久久久麻豆精品 | 91视频a| 国产一区精品在线 | 六月婷婷久香在线视频 | 国产精品美女久久久网av | 日本亚洲国产 | 欧美特一级片 | 日韩在线中文字幕视频 | 久久这里精品视频 | 久久综合网色—综合色88 | 久久久久久国产一区二区三区 | 日韩欧美在线观看一区二区 | 久久在线 | 久久久精品国产一区二区电影四季 | 99久久精品国产毛片 | 亚洲资源在线网 | 久久草网站 | 欧美精品午夜 | 日韩精品一区二区在线视频 | 国产视频手机在线 | 91三级在线观看 | 国产视频在线播放 | 涩涩色亚洲一区 | 久免费视频 | 99在线免费观看 | 国产黄免费 | 免费在线观看亚洲视频 | 欧美成人高清 | 免费黄色av | 91视频午夜 | 天天爽天天摸 | 国产一区二区三区高清播放 | 97av在线视频免费播放 | 国产在线观看 | 7777xxxx| 久久99国产精品二区护士 | 日产乱码一二三区别在线 | 欧美 另类 交 | 99久久er热在这里只有精品66 | 欧美国产日韩在线视频 | 午夜精品久久久久久久99婷婷 | 国产一区二区三区高清播放 | 香蕉影视| aav在线| 日日摸日日添夜夜爽97 | 91插插插网站 | 人人干人人干人人干 | 久久精品www人人爽人人 | 麻豆 91 在线 | 国产xxxx做受性欧美88 | 91亚洲精| 九九在线免费视频 | 亚洲香蕉在线观看 | 四虎影视成人永久免费观看视频 | 国产精品久久久久免费a∨ 欧美一级性生活片 | 国产特级毛片aaaaaaa高清 | 精品久久久影院 | 人人狠狠综合久久亚洲 | 欧美片一区二区三区 | 六月婷婷久香在线视频 | 国产中文字幕第一页 | 成年人app网址 | 99精品区 | 五月天亚洲综合 | 中文字幕制服丝袜av久久 | 亚洲欧美偷拍另类 | 精品女同一区二区三区在线观看 | 99这里只有精品视频 | 精品久久网站 | 911亚洲精品第一 | 国产丝袜高跟 | 国产综合婷婷 | 91传媒91久久久 | 亚洲精品一区二区三区高潮 | 99久久婷婷国产综合亚洲 | a天堂中文在线 | 久久久美女 | 视频一区在线播放 | 美女视频黄是免费的 | 伊人首页| 福利视频一区二区 | 日韩啪啪小视频 | 久久久午夜剧场 | 成 人 免费 黄 色 视频 | 91视频免费看 | 中文字幕丝袜 | 天天干天天草 | 国产精品视频专区 | 波多野结衣在线视频免费观看 | 人人玩人人爽 | 精品一区二区av | 97香蕉久久超级碰碰高清版 | www.看片网站| 亚洲国产精品va在线看黑人动漫 | 日韩久久久久久久久久久久 | www激情久久 | 国产手机在线播放 | 亚洲精品成人在线 | 国产精品久久久电影 | 久久精品9 | 久久久黄色免费网站 | 成人一区二区三区在线观看 | 一区国产精品 | 天天干天天操天天拍 | 97av影院 | 色资源在线 | 婷五月激情 | 亚洲视频久久 | 日韩精品短视频 | 狠狠躁日日躁 | 欧美日韩免费在线观看视频 | 中文字幕一区二区三区在线播放 | 天天操天天弄 | 欧美日韩国产亚洲乱码字幕 | 亚洲经典视频在线观看 | 久久夜色电影 | 国产一区精品在线 | 草樱av| 久久与婷婷 | 97免费在线观看视频 | av免费在线网站 | 久久久免费看视频 | 国产成人免费在线 | 偷拍福利视频一区二区三区 | 婷婷国产视频 | 久久免费观看少妇a级毛片 久久久久成人免费 | 亚洲三级毛片 | 玖玖色在线观看 | 精品视频99 | 亚洲综合丁香 | 日韩欧美视频免费观看 | 国产在线国产 | 欧美日韩在线观看一区二区三区 | 色综合天天视频在线观看 | 99色在线 | h视频日本 | 色先锋资源网 | 成人午夜电影网 | 国产精品一区二区 91 | 欧美日韩精品在线观看 | 91精品国产成人 | 又黄又刺激视频 | 亚洲理论在线 | 91精品国产综合久久婷婷香蕉 | 中文字幕在线免费播放 | 精品资源在线 | 亚洲精品av中文字幕在线在线 | 日韩在线免费视频 | 91视频3p | 日韩精品视| 日本久久久精品视频 | 久草在线免费新视频 | 国产精品九九九九九九 | 午夜精品久久久久久 | 男女啪啪免费网站 | 人人澡人人舔 | 青青草久草在线 | 天天草天天摸 | aav在线 | 久久国产精品偷 | 日韩精品一区二 | 欧美日韩色婷婷 | 18女毛片| zzijzzij亚洲成熟少妇 | 1000部国产精品成人观看 | 99热在线看 | 在线播放 一区 | 少妇搡bbb | 免费a v视频 | 午夜视频在线观看一区二区 | 日韩精品一区二区三区视频播放 | 欧美一级片在线 | www色,com| 97av色| 国产成免费视频 | 日韩国产精品久久 | 99re亚洲国产精品 | 美女视频永久黄网站免费观看国产 | 91免费国产在线观看 | 国内小视频 | 女人久久久久 | 久久免费的精品国产v∧ | 色综合久久精品 | 亚洲最大的av网站 | 日本黄色免费大片 | 久草视频中文 | 欧美成人理伦片 | 欧美日韩精品在线免费观看 | 亚洲精品视频在线观看网站 | 97视频总站| 成人黄色毛片视频 | 成人av资源网 | 亚洲电影在线看 | 亚洲成人蜜桃 | www.777奇米| 伊人网综合在线观看 | 亚洲视频在线免费观看 | 国产在线污 | 丁香六月综合网 | 不卡的av在线播放 | 国产成人久久精品亚洲 | 亚洲精品久久久久中文字幕二区 | 久久草 | 毛片888| 色视频在线观看免费 | 国产一级片毛片 | 欧美日韩成人 | 日韩a免费| 国产精品第一页在线观看 | 五月天综合 | 黄色特级片 | 亚洲全部视频 | 一区二区三区免费在线观看视频 | 99久久视频| 日韩免费视频在线观看 | 丁香5月婷婷久久 | 亚洲欧美国产视频 | 久久综合影院 | av先锋影音少妇 | 日韩精品中文字幕有码 | 激情五月婷婷激情 | 黄色在线观看免费网站 | 亚洲成人精品久久久 | 7777精品伊人久久久大香线蕉 | av在线免费在线观看 | 日韩中文字幕免费看 | 成人在线观看av | 天天爽综合网 | 国产精品免费麻豆入口 | 色综合亚洲精品激情狠狠 | 国产精品久久久久国产精品日日 | www一起操| 亚洲精品美女视频 | 天天摸天天弄 | 国产精品美 | 久久精品一区二区 | 人人爽人人爽人人片 | 国产午夜激情视频 | 激情av在线资源 | av电影在线观看完整版一区二区 | 日韩成人一级大片 | 毛片区 | 免费观看一区 | 国产精品毛片久久久久久 | 在线免费精品视频 | 国产日韩欧美在线观看 | 狠狠色伊人亚洲综合成人 | 二区视频在线观看 | 欧美日韩不卡一区 | 久草热久草视频 | 区一区二区三区中文字幕 | 国产97免费| 国产黄色免费电影 | 久久精品美女 | 国产精品成人a免费观看 | 丝袜av网站 | 免费观看www7722午夜电影 | 天天综合网天天综合色 | 在线欧美小视频 | 色99中文字幕 | 欧美日韩久 | 超碰在线人人艹 | 国产在线97 | 狠狠色狠狠色综合系列 | 成人禁用看黄a在线 | 久久深夜福利免费观看 | 啪啪精品| 欧美一级电影免费观看 | 国产香蕉av | 在线免费高清一区二区三区 | 最新国产精品拍自在线播放 | 日韩在线免费播放 | 黄色网址中文字幕 | 欧美日韩国产一区二区三区 | 国产不卡视频 | 六月丁香激情综合 | 欧美一级性生活 | 99热官网 | 成人97人人超碰人人99 | 国产中文字幕视频 | 中文字幕国产 | 欧美日韩免费一区二区三区 | 国产呻吟在线 | 日韩久久精品一区 | 97精品久久人人爽人人爽 | 久久久久免费视频 | 国产午夜一区二区 | 女人高潮一级片 | 欧美日韩高清一区二区三区 | 国产一区二区视频在线播放 | 91欧美精品 | 在线中文字幕视频 | 草久久久久| 麻花豆传媒mv在线观看 | 中文字幕精品一区久久久久 | 午夜视频免费播放 | 国产高清成人在线 | 91免费的视频在线播放 | 国产糖心vlog在线观看 | 最近中文字幕国语免费高清6 | 久久国语 | 中文字幕一区二区三区四区久久 | 国产精品3区 | 中文字幕永久 | 一级欧美黄 | 天天操天天干天天爱 | 六月丁香在线观看 | 特级a毛片 | 成人全视频免费观看在线看 | 久久99精品国产麻豆宅宅 | 亚洲精品视频在线免费播放 | 久久久一本精品99久久精品 | 色噜噜日韩精品一区二区三区视频 | 久久观看免费视频 | 久久综合狠狠综合久久狠狠色综合 | 国产精品视频999 | 懂色av懂色av粉嫩av分享吧 | 成人性生交大片免费看中文网站 | 三级免费黄 | 亚洲最新av在线网站 | 国产精品精品久久久久久 | 国产无吗一区二区三区在线欢 | 欧美精品在线视频 | 日韩草比| 精品一区二区三区电影 | 最新中文字幕在线观看视频 | 韩日av在线 | 久久资源在线 | 91成人免费电影 | 91香蕉视频黄色 | 在线观看免费中文字幕 | 欧美成人性网 | 欧美狠狠色 | 欧美另类调教 | 成人午夜电影在线播放 | 国产香蕉视频在线播放 | 夜夜夜影院 | 天天躁天天狠天天透 | 国产亚洲精品成人av久久影院 | 亚洲黄色免费电影 | 欧美日韩视频在线 | 午夜av在线免费 | 91日韩在线专区 | 日韩av男人的天堂 | 久草在线视频免费资源观看 | 国产精品久久久久久欧美 | 国产黄色a| 一区二区三区日韩视频在线观看 | 亚洲成人精品在线观看 | 超碰在线中文字幕 | 亚洲欧美怡红院 | 国产高清在线一区 | 欧美伦理一区 | 一区二区三区 中文字幕 | 欧亚日韩精品一区二区在线 | 国产原厂视频在线观看 | 免费网站在线观看人 | 美女黄频在线观看 | 91成人精品 | 亚洲va韩国va欧美va精四季 | 黄色av三级在线 | 麻豆超碰| av手机在线播放 | 成年人免费看 | av天天在线观看 | 国产在线一区观看 | 免费三级av | 一区二区三区在线免费观看视频 | 在线va网站 | 麻豆国产精品va在线观看不卡 | 最新影院 | 97小视频| 日日草天天草 | 日韩在线在线 | 国产精品美女久久久久久久 | 久久国产精品一区二区三区四区 | 国产免费专区 | 在线影院av | 精品国产诱惑 | 91九色最新地址 | 99久久精品久久久久久动态片 | www夜夜操com | 天天综合视频在线观看 | 黄色网址中文字幕 | 91精品国产九九九久久久亚洲 | 精品国产网址 | 天天操网址 | 午夜精品av | av片一区二区| 欧美另类xxxx| 最近2019中文免费高清视频观看www99 | 亚洲 欧洲 国产 日本 综合 | 亚洲国内在线 | 玖草影院 | 国产三级久久久 | 99久久国产免费免费 | 99re久久资源最新地址 | 狠狠狠狠狠狠狠狠干 | 深爱激情av | 欧美日韩网站 | 粉嫩av一区二区三区四区五区 | 欧洲精品二区 | 亚洲日韩中文字幕 | 国产在线精 | 一本色道久久综合亚洲二区三区 | 在线成人高清电影 | 高潮毛片无遮挡高清免费 | 在线看小早川怜子av | 国产精品久久久久久久久费观看 | 日韩精品亚洲专区在线观看 | 日韩视频中文字幕 | 日韩一级黄色大片 | 亚洲精品观看 | 91系列在线| 六月久久婷婷 | 日日夜夜精品免费 | 国产又粗又猛又黄又爽 | 午夜婷婷综合 | 国产99久 | av片在线看 | 久久国内精品视频 | 人人爽人人看 | 青草视频在线播放 | 九九热re| 亚洲精品国产精品国 | 一本大道久久精品懂色aⅴ 五月婷社区 | 国产午夜三级一区二区三桃花影视 | 国产美女精品视频免费观看 | 2021国产精品| 午夜av一区 | 久在线观看 | 青草视频在线播放 | 人人爽人人爽人人爽 | 精品一区二区精品 | 国产免费二区 | 国产特级毛片aaaaaa高清 | 久久久午夜精品福利内容 | 中文字幕高清在线播放 | 视频福利在线 | 黄色av大片 | 国产护士在线 | 操夜夜操| 成人一区二区三区在线 | 欧美成人手机版 | 麻豆免费在线播放 | 欧美激情xxxx | 国产精品手机视频 | 亚洲欧美精品一区 | 色资源中文字幕 | 狠狠色丁香婷婷综合久久片 | 超碰人人国产 | 亚洲免费色 | 免费观看的黄色片 | 日本不卡123区 | 成人av影视观看 | 欧美成人基地 | 九九热只有精品 | 精品欧美乱码久久久久久 | 涩涩网站在线播放 | 黄色亚洲精品 | 91精品国产92久久久久 | 国产中文字幕一区 | 日韩视频一区二区在线 | 黄色av网站在线观看 | 免费看黄的视频 | 久久免费视频网站 | 久久色视频| 国产精品久久久久一区二区三区共 | 99热日本| 国产69精品久久久久久 | 欧美在线日韩在线 | 人人澡人人模 | 日韩精品一区二区免费视频 | 日本在线观看一区二区 | 国产中文字幕91 | 亚洲少妇久久 | 亚洲视频专区在线 | 日本中文一级片 | 免费久久网站 | 欧美日韩免费一区二区三区 | 国产网站在线免费观看 | 美女视频免费精品 | 国内精品久久久久久久影视麻豆 | 日韩欧美在线视频一区二区三区 | 麻豆影视在线播放 | 香蕉视频网址 | 在线观看网站av | av高清免费 | 在线观看免费高清视频大全追剧 | 免费在线观看黄网站 | 久久婷亚洲五月一区天天躁 | 久草视频在线免费播放 | 久久久久色 | 久久久久久免费毛片精品 | 激情黄色一级片 | 国产精品一区二区 91 | 成人在线观看网址 | 97视频在线观看视频免费视频 | 亚洲精品一区二区三区四区高清 | 一级片视频在线 | 久久精品99精品国产香蕉 | 久久久久久久久久久精 | 婷婷色综| 亚洲欧美日韩不卡 | 国产高清在线一区 | 久久久免费网站 | 亚洲综合色丁香婷婷六月图片 | 色开心 | 国产精品久久一区二区无卡 | 天天操天天操天天操天天操 | 人人干人人干人人干 | 91视频久久久 | 97av影院 | 在线小视频你懂得 | 久久天天躁狠狠躁夜夜不卡公司 | 日韩av片无码一区二区不卡电影 | 免费网站看av片 | 中文字幕在线视频免费播放 | 久久99精品国产99久久 | 中文字幕日韩精品有码视频 | 天天操夜夜看 | 亚洲天堂首页 | 国产高清在线a视频大全 | 看全黄大色黄大片 | 色综合久久综合网 | 五月视频 | 婷婷色 亚洲 | www.夜夜操.com | 中文字幕在线精品 | 欧美日韩国产一区二区三区在线观看 | 精品综合久久久 | 黄色网在线免费观看 | 中文字幕av日韩 | 国产精品入口久久 | 午夜精品福利一区二区 | 精品久久久久久久久久久久久久久久久久 | 99热在线观看免费 | 日韩av在线看 | 在线观看免费视频 | 五月天综合婷婷 | 人人射 | 婷婷夜夜 | 99久久久国产精品美女 | av网站免费线看精品 | 中文字幕第一页在线视频 | 久久这里 | 欧美色图88 | 国产91全国探花系列在线播放 | 99精品视频在线观看免费 | 中文字幕在线高清 | 久久蜜桃av | 日本三级久久久 | www.com久久 | 人人插人人搞 | 午夜黄色影院 | 国产日本在线观看 | 91九色免费视频 | 欧美91精品久久久久国产性生爱 | 国产精品资源在线观看 | 国产精品毛片一区二区 | 亚洲视频电影在线 | 国产高清在线a视频大全 | 特级a老妇做爰全过程 | 免费男女羞羞的视频网站中文字幕 | 欧美日韩在线精品 | 夜色成人av | 在线日本看片免费人成视久网 | 免费日韩三级 | 午夜影院一区 | 99精品视频在线看 | 中文一区二区三区在线观看 | www日韩视频| 久久精品电影 | www.天天射| 乱男乱女www7788 | 99久久99热这里只有精品 | 精品久久久亚洲 | 亚洲精品在线免费看 | 国产一级免费视频 | 色鬼综合网 | 欧女人精69xxxxxx | 国产区在线看 | 国产经典 欧美精品 | 在线观看成人福利 | 丁香影院在线 | 婷婷久久精品 | 黄污在线看 | www.色午夜,com | 国产精品影音先锋 | 91成人久久 | 97超碰在线久草超碰在线观看 | 亚洲成人免费观看 | 亚洲综合导航 | 激情亚洲综合在线 | 中文字幕日韩免费视频 | 91精品1区| 四虎免费在线观看视频 | 精品久久精品 | 国产香蕉97碰碰久久人人 | 亚洲人在线7777777精品 | 国产成人黄色 | 亚洲免费一级电影 | 国产馆在线播放 | 亚洲色图美腿丝袜 | 有码视频在线观看 | 伊人天天狠天天添日日拍 | 国产又粗又猛又爽又黄的视频先 | 综合久久久 | 丁香六月婷婷 | 久久久在线观看 | 日韩欧美综合视频 | 久久精品视频中文字幕 | 人人干狠狠干 | 一区二三国产 | 欧美性久久久 | 日韩大片在线看 | 精品黄色片 | 国产精品麻豆三级一区视频 | 国产91av视频在线观看 | 国产精品免费观看视频 | 国产精品毛片网 | 国产精品第2页 | 国产破处精品 | 免费在线观看国产精品 | 欧美日韩午夜 | 天堂在线免费视频 | 天天色天天色天天色 | 中文字幕字幕中文 | 5月丁香婷婷综合 | 91观看视频| 色午夜影院 | 欧美人体xx | 免费男女羞羞的视频网站中文字幕 | 国产不卡高清 | 国产精品久久久久久久久久免费看 | 亚洲欧美日韩一二三区 | 国内揄拍国内精品 | 国产一性一爱一乱一交 | 亚洲综合在 | 久久艹久久 | 一本一本久久a久久精品综合小说 | 免费麻豆 | 一区二区视频在线免费观看 | 黄色一级在线观看 | 天天操天天摸天天射 | 黄色美女免费网站 | 亚洲精品777 | 精品亚洲网 | 日本久久久久 | 中文字幕免费高清av | 欧美调教网站 | 精品999在线观看 | 中文字幕av免费观看 | 久久久久久久久久久久久影院 | 久久久久国产一区二区三区四区 | 成年人在线视频观看 | 国产黄网站在线观看 | 亚洲一级理论片 | 中文字幕亚洲情99在线 | 成人欧美亚洲 | 精品中文字幕在线观看 | 精品国产一区二区三区av性色 | 综合激情av| 亚洲久草在线 | 91精品在线观看视频 | 久久久久免费精品视频 | 国产精品入口66mio女同 | 99久久精品网 | 亚洲成人精品久久 | 国产精品免费观看国产网曝瓜 | 婷婷日韩 | 婷婷色中文 | 国产123av| 成人a免费看 | 久久少妇免费视频 | 久久免费播放 | 欧美日韩在线播放 | a级国产乱理论片在线观看 伊人宗合网 | 最近中文字幕完整视频高清1 | 色偷偷888欧美精品久久久 | 97香蕉超级碰碰久久免费软件 | 日本精品一区二区三区在线观看 | 欧美在线free | 操操操av| 天天摸夜夜添 | 国产99久久久国产精品免费二区 | 久久99亚洲精品久久 | 国产成人精品一区二 | 久久爽久久爽久久av东京爽 | 69av免费视频 | 日韩videos| 成人免费视频在线观看 | 国产中文欧美日韩在线 | 欧美一级片在线免费观看 | 91在线超碰 | 国产免费观看久久黄 | 国产精品成人a免费观看 | 日韩精品一区在线播放 | 91av播放 | 免费看一级一片 | 国产亚洲精品中文字幕 | 99精品欧美一区二区三区黑人哦 | 日日夜夜骑| 日韩欧美在线观看一区二区三区 | 91成人小视频 | 久久天天躁狠狠躁夜夜不卡公司 | 亚洲国产精品人久久电影 | 91亚洲精品久久久中文字幕 | 亚洲精品视频中文字幕 | 天天色影院 | 日韩免费在线 | 人人要人人澡人人爽人人dvd | 免费观看成人av | 精品国产一区二区三区四区在线观看 | 激情综合一区 | 国产成人一区二区三区免费看 | 欧美日韩在线观看一区二区三区 | 一二三区视频在线 | 亚洲精品美女久久17c | 成人综合日日夜夜 | 中文字幕观看在线 | 亚洲欧美日本一区二区三区 | 激情视频久久 | 国产 色| 国产va饥渴难耐女保洁员在线观看 | 色婷丁香| 日韩综合色 | 天天射天天色天天干 | 亚洲狠狠操 | www国产亚洲精品久久麻豆 | 毛片网在线| 欧美日本一二三 | 992tv又爽又黄的免费视频 | 国产午夜精品av一区二区 | 亚洲美女免费视频 | 九九热免费在线视频 | 中文字幕在线视频免费播放 | 欧美最猛性xxxxx免费 | 色多多在线观看 | 国产视频在线播放 | 99久久精品无码一区二区毛片 | 欧美日韩国产精品久久 | 国产成人免费 | av一级一片 | 蜜臀久久99精品久久久无需会员 | 精品国产aⅴ麻豆 | 91精品一区二区三区久久久久久 | 婷婷色网站 | 在线免费观看国产 | 精品国产三级a∨在线欧美 免费一级片在线观看 | 国产三级av在线 | 国产黄免费在线观看 | 国产激情久久久 | 91精品在线观看视频 | 国产精品免费观看国产网曝瓜 | 日本特黄一级 | 免费亚洲成人 | 欧美精品一级视频 | 成人在线观看网址 | 亚洲精品自拍视频在线观看 | 亚洲另类视频 | 欧美色综合久久 | 免费精品视频在线 | 黄色小说免费观看 | 久久一级电影 | 天天干亚洲 | 亚洲国产午夜视频 | 亚洲国产欧美在线人成大黄瓜 | 日韩欧美国产精品 | 国产精品一区二区久久久久 | 日韩v欧美v日本v亚洲v国产v | 亚洲黄色网络 | 欧美视屏一区二区 | 99中文在线| 丁香六月五月婷婷 | 久久综合九九 | 国产黄在线免费观看 | 久久久久久久久久久久久国产精品 | 91黄视频在线观看 | 77国产精品 | 久久精品国产一区二区三区 | 激情婷婷在线观看 | 亚洲va欧美va人人爽春色影视 | 国产午夜精品福利视频 | 狠狠色伊人亚洲综合成人 | www.五月婷婷 | 蜜臀av夜夜澡人人爽人人 | 手机av在线网站 | 五月婷激情| 亚洲视频久久 | 久久99国产精品免费 | 日韩91av| 中文字幕欧美日韩va免费视频 | 国产精品综合久久久久久 | 国产精品久久久久免费 | 日韩草比| 中文字幕视频一区 | 久草视频免费 | 欧美日韩精品免费观看视频 | 国产你懂的在线 | 欧美精品三级 | 五月婷婷.com| 五月天丁香综合 | 久久亚洲专区 | 久久精品视频在线 | 成年美女黄网站色大片免费看 | 日韩高清在线一区二区 | 九九色在线观看 | 亚洲第一中文字幕 | 在线观看完整版 | 国产精品18久久久久久久久 | 999日韩 | 国产伦理精品一区二区 | 中文字幕色婷婷在线视频 | 国产精品免费久久久久 | 国产 在线 高清 精品 | 国产亚洲精品中文字幕 | 天天爱天天草 | 日韩欧美在线一区 | 国产午夜精品理论片在线 | www五月天| 久久黄色免费观看 | 人人爽人人澡人人添人人人人 | 天天综合久久综合 | 成人av电影免费 | 久久久久免费精品 | 久久99最新地址 | 日韩在线电影一区二区 | 毛片永久新网址首页 | 色噜噜噜 | 国产成人免费观看久久久 | 在线观看免费视频 | 欧美日韩精品电影 | 色丁香婷婷| 欧美日韩国产在线一区 | 日本黄区免费视频观看 | 一本一本久久a久久精品牛牛影视 | 中文字幕免费一区二区 | 看片网站黄 | av免费播放 | 啪啪精品| 日日草天天干 | 国产一区二区三区久久久 | 日韩专区av | 免费日韩高清 | 91亚洲精品久久久中文字幕 | 日韩视频一二三区 | 日日夜夜天天射 | 日韩免费看视频 | 欧洲精品久久久久毛片完整版 | 91av网址 | 久久久www成人免费毛片麻豆 | 在线欧美中文字幕 | 九月婷婷人人澡人人添人人爽 | 国内精品久久久久影院日本资源 | 日韩免费看的电影 | 中文av在线免费观看 | 97福利视频 | av丝袜在线 | 中文字幕亚洲综合久久五月天色无吗'' | 黄毛片在线观看 | 在线色吧 | 久久精品99国产 | 精品免费国产一区二区三区四区 | 亚洲涩涩网 | 天天草天天| 日韩一级片观看 | 国产免费xvideos视频入口 | 成人性生交大片免费看中文网站 | 日韩欧美在线播放 | 日日躁你夜夜躁你av蜜 | 日本中文乱码卡一卡二新区 | 97品白浆高清久久久久久 | 96超碰在线 | 欧美一级片免费观看 | 国产美女久久 | 国产精品视频免费看 | 最近2019中文免费高清视频观看www99 | 久草视频在线观 | 91热这里只有精品 | 日韩高清精品免费观看 | 西西444www大胆高清视频 | 青春草免费在线视频 | 高清一区二区三区 | 国内精品小视频 | 中文字幕在线观看视频免费 | 91麻豆看国产在线紧急地址 | 欧美日韩在线视频一区二区 | 操操操日日 | 久久综合色婷婷 | 成年人av在线播放 | 久久国产免费视频 | 少妇bbbb搡bbbb搡bbbb | www.黄色小说.com | 玖玖在线观看视频 | 国产精品video爽爽爽爽 | 狠狠的操狠狠的干 | 欧美精品久久人人躁人人爽 | 在线观看日韩中文字幕 | 69视频永久免费观看 | 香蕉在线观看视频 | 精品国产成人 | 欧美性黑人| 精品视频在线观看 | 国产日韩中文字幕 | 免费人成网ww44kk44 | 99久久夜色精品国产亚洲96 | 91亚洲精品久久久久图片蜜桃 | 成人国产一区二区 | 97久久精品午夜一区二区 | 国产91精品一区二区绿帽 | 深爱婷婷网| 丁香婷婷在线 |