树莓派小车折腾记

买了二手树莓派4B大半年,想玩个小车。
看了淘宝,不带板子要350一个。
淘宝树莓派小车
自己淘点便宜材料:

  1. 四轮车底盘+4个电机:21.6元
  2. L298N驱动板2个:5.5元x2
  3. 18650锂电池4节+2个电池盒:22元
  4. 电烙铁+焊锡:10元
  5. 导线:2元
  6. 各种杜邦线:2元
  7. HX1838遥控器:4元
  8. 树莓派摄像头:19元
  9. 10000mAH小米充电宝:旧的

装好了:
一开始想搞成Network RC 网络摄像头遥控赛车
一开始想搞成Network RC 网络摄像头遥控赛车
Network RC:https://network-rc.esonwong.com/
想想还是算了,还得弄个旧手机做热点,还是遥控器操作吧。
装上天线
小朋友楼下玩了十几分钟,HX1838的接收器就烧了
小朋友楼下玩了十几分钟,HX1838的接收器就烧了。
也不知道咋回事。
突然听到传言因为缺芯,树莓派的价格已经涨上天了。
去淘宝瞅瞅,这个二手400块买的树莓派4B 4GB已经涨到一千多了。
装遥控车上有点大材小用,正好遥控坏了。又下单买了

  1. 树莓派Pico:30元
  2. PS2遥控器+接收器:40元

装好了:

冷知识:充电宝给树莓派PICO供电30s会自动停止,推测是充电宝小电流自动关机功能导致,有的充电宝可以连续点击2次开关,关闭小电流保护功能

用树莓派Pico时踩了一个坑:
在main.py中写了一个死循环之后,再连电脑就不能进入调试了,会出现Device is busy or does not respond。
需要下载 闪存重置文件,按住板子上 bootsel 键,通电后, 放置该文件进去, 清空闪存之后,再刷入micropython固件即可完成重置。
闪存重置文件:https://datasheets.raspberrypi.com/soft/flash_nuke.uf2
小朋友再玩下
祝汪宝第四个儿童节快乐

最后贴一个 PS2手柄+使用Micropython
https://www.jianshu.com/p/30723e5624ae
亲测第二个可用,改几个GPIO接口编号就行,
貌似也是从Github搬运:
https://github.com/shallwe/micropython_ps2

import time
from machine import Pin

class PS2Controller:
    # These are our button constants
    SELECT = 1
    L3 = 2
    R3 = 3
    START = 4
    UP = 5
    RIGHT = 6
    DOWN = 7
    LEFT = 8
    L2 = 9
    R2 = 10
    L1 = 11
    R1 = 12
    TRIANGLE = 13
    CIRCLE = 14
    CROSS = 15
    SQUARE = 16
    KEYS = dict([
                  (SELECT, "SELECT"),        (L3, "L3"),        (R3, "R3"),        (START, "START"),
                  (UP, "UP"),        (RIGHT, "RIGHT"),        (DOWN, "DOWN"),        (LEFT, "LEFT"),
                  (L2, "L2"),        (R2, "R2"),        (L1, "L1"),        (R1, "R1"),
                  (TRIANGLE, "TRIANGLE"),        (CIRCLE, "CIRCLE"),        (CROSS, "CROSS"),        (SQUARE, "SQUARE")  ])

    CTRL_CLK = 10
    CTRL_BYTE_DELAY = 16

    CMD_SHORT_POLL   = [0x01, 0x42, 0x00, 0x00, 0x00]
    CMD_ENTER_CONFIG = [0x01, 0x43, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00]
    CMD_SET_MODE     = [0X01, 0x44, 0x00,
                        0x01,  # 00 normal; 01 red or analog
                        0x03,  # 03 lock; ee no lock
                        0x00, 0x00, 0x00, 0x00]
    CMD_SET_BYTES_LARGE = [0x01, 0x4F, 0x00, 0xFF, 0xFF, 0x03, 0x00, 0x00, 0x00]
    CMD_EXIT_CONFIG     = [0x01, 0x43, 0x00, 0x00, 0x5A, 0x5A, 0x5A, 0x5A, 0x5A]
    CMD_ENABLE_RUMBLE   = [0x01, 0x4D, 0x00, 0x00, 0x01]
    CMD_TYPE_READ       = [0x01, 0x45, 0x00, 0x5A, 0x5A, 0x5A, 0x5A, 0x5A, 0x5A]
    CMD_READ_DATA       = [0X01, 0X42, 0X00, 0X00, 0X00, 0X00, 0X00, 0X00, 0X00]

    MAX_READ_DELAY = 1500

    VALID_MODES = [0X41, 0X73]

    def __init__(self, di_pin_no=26, do_pin_no=27, cs_pin_no=14, clk_pin_no=12):  # DI=DAT、DO=CMD 可以在此处将针脚调整为你自己对应的针脚
        self.di_pin_no = di_pin_no
        self.do_pin_no = do_pin_no
        self.cs_pin_no = cs_pin_no
        self.clk_pin_no = clk_pin_no
        self.di  = Pin(self.di_pin_no, Pin.IN)  # DI = DAT
        self.do  = Pin(self.do_pin_no, Pin.OUT) # DO = CMD
        self.cs  = Pin(self.cs_pin_no, Pin.OUT)
        self.clk = Pin(self.clk_pin_no, Pin.OUT)
        self.buff_out = [0x01, 0x42]
        self.buff_in = [0] * 9
        self.pressed_keys = []
        self.read_delay = 1
        self.last_read_ms = 0
        self.lx = 0
        self.ly = 0
        self.rx = 0
        self.ry = 0

    @property
    def red_mode(self):  # analog mode
        return self.buff_in[1] & 0xf0 == 0x70

    def do_h(self):
        self.do.value(1)

    def do_l(self):
        self.do.value(0)

    def cs_h(self):
        self.cs.value(1)

    def cs_l(self):
        self.cs.value(0)

    def clk_h(self):
        self.clk.value(1)

    def clk_l(self):
        self.clk.value(0)


    # noinspection PyUnresolvedReferences
    def delay_byte(self):
        time.sleep_us(self.read_delay)

    # noinspection PyUnresolvedReferences
    def delay_clk(self):
        time.sleep_us(self.CTRL_CLK)

    # noinspection PyUnresolvedReferences
    def delay_read(self):
        time.sleep_us(self.CTRL_BYTE_DELAY)
        # time.sleep_ms(self.read_delay)

    def cmd(self, cmd):
        ret = 0
        for i in range(8):
            if cmd & 1 << i:
                self.do_h()
            else:
                self.do_l()
            self.clk_l()
            self.delay_clk()
            if self.di.value():
                ret |= 1 << i
            self.clk_h()
        self.do_h()
        self.delay_byte()
        return ret

    """
    :param 
    pure True means not delay
    """

    def cmd_group(self, cmds):
        self.cs_l()
        self.delay_byte()
        for cmd in cmds:
            self.cmd(cmd)
        self.cs_h()
        self.delay_read()

    def init(self):
        self.di  = Pin(self.di_pin_no, Pin.IN)
        self.do  = Pin(self.do_pin_no, Pin.OUT)
        self.cs  = Pin(self.cs_pin_no, Pin.OUT)
        self.clk = Pin(self.clk_pin_no, Pin.OUT)
        self.do_h()
        self.clk_h()

        self.read_once()
        self.read_once()
        #
        if self.buff_in[1] not in [0x41, 0x73]:
            #print("control type not ok, expect 41 73 79, bug get ", "{:02x}".format(self.buff_in[1]))
            return 1

        self.read_delay = 1

        for i in range(10):
            # self.cmd_group(self.CMD_SHORT_POLL)
            # self.cmd_group(self.CMD_SHORT_POLL)
            # self.cmd_group(self.CMD_SHORT_POLL)
            self.cmd_group(self.CMD_ENTER_CONFIG)
            self.delay_byte()

            self.do_h()
            self.clk_h()
            self.cs_l()

            self.delay_byte()
            #
            temp = [0] * len(self.CMD_TYPE_READ)
            for j in range(9):
                for cmd in self.CMD_TYPE_READ:
                    temp[j] = self.cmd(cmd)
            self.cs_h()

            self.cmd_group(self.CMD_SET_MODE)
            # self.cmd_group(self.CMD_ENABLE_RUMBLE)
            self.cmd_group(self.CMD_EXIT_CONFIG)
            self.read_once()
            if self.buff_in[1] in self.VALID_MODES:
                #print("read_delay configed,", self.read_delay)
                break
            else:
                self.read_delay += 1
                #print("read_delay++,", self.read_delay)

    def reconfig(self):
        #print("reconfig")
        self.cmd_group(self.CMD_ENTER_CONFIG)
        self.cmd_group(self.CMD_SET_MODE)
        self.cmd_group(self.CMD_EXIT_CONFIG)

    def p(self, debug=True):
        if debug:
            for d in self.buff_in:
                print("{:08b}".format(d))

        key_raw = (self.buff_in[4] << 8) | self.buff_in[3]
        for i in range(1, 17):
            if not key_raw & 1 << i - 1:
                self.pressed_keys.append(i)

        if self.red_mode:
            self.rx = self.buff_in[5] - 128
            self.ry = self.buff_in[6] - 128
            self.lx = self.buff_in[7] - 128
            self.ly = self.buff_in[8] - 128
        if self.pressed_keys:
            out = "keys:" + ','.join(self.KEYS[k] for k in self.pressed_keys) + "; "
        else:
            out = ""
        # for key in self.pressed_keys:
        #     print(key, self.KEYS[key])
        if self.red_mode and (out or any(x != 0 for x in [self.rx, self.ry, self.lx, self.ly])):
            out += "pos: (lx,ly):{},{};  (rx,ry): {},{}".format(self.lx, self.ly, self.rx, self.ry)

        if out:
            print(out)

    def read_once(self, debug=False):
        now = time.ticks_ms()
        delay = now - self.last_read_ms
        if delay > self.MAX_READ_DELAY:
            #print(now, self.last_read_ms, delay)
            self.reconfig()
        elif delay < self.read_delay:
            # noinspection PyUnresolvedReferences
            time.sleep_ms(self.read_delay - delay)

        self.buff_in = [0] * 9
        self.pressed_keys.clear()

        for j in range(5):
            # for i in range(1):
            self.do_h()
            self.clk_h()
            self.cs_l()
            self.delay_byte()

            for i, c in enumerate(self.CMD_READ_DATA):
                self.buff_in[i] = self.cmd(c)

            self.cs_h()

            if self.buff_in[1] in self.VALID_MODES:
                break
            else:
                #print("mode: {:08b}, retry_J: {}".format(self.buff_in[1], j))
                self.reconfig()
                self.delay_read()
        if self.buff_in[1] not in self.VALID_MODES and self.read_delay < 10:
            self.read_delay += 1

        self.last_read_ms = time.ticks_ms()
        #self.p(debug)

        #以下为自己添加 ,原P()内的代码
        key_raw = (self.buff_in[4] << 8) | self.buff_in[3]
        for i in range(1, 17):
            if not key_raw & 1 << i - 1:
                self.pressed_keys.append(i)

        if self.red_mode:
            self.rx = self.buff_in[5] - 128
            self.ry = self.buff_in[6] - 128
            self.lx = self.buff_in[7] - 128
            self.ly = self.buff_in[8] - 128
        if self.pressed_keys:      #注意下面这几行的:和,千万不要修改,后面要他们做标记分割使用!
            out = "keys:" + ','.join(self.KEYS[k] for k in self.pressed_keys) + ": "
        else:
            out = "keys:无:"
        for key in self.pressed_keys:
             print(key, self.KEYS[key])
        '''
        if self.red_mode and (out or any(x != 0 for x in [self.rx, self.ry, self.lx, self.ly])):
            out += "pos(lx,ly):{},{}:  pos(rx,ry):{},{}:".format(self.lx, self.ly, self.rx, self.ry)
        if out:
            print(out)
        '''    
        #return self.buff_in
        return out

'''
# 调用方法,将以下几行代码新建一个文件如:ps2_test.py
from ps2 import PS2Controller
import time
ps2ctl = PS2Controller(di_pin_no=26, do_pin_no=27, cs_pin_no=14, clk_pin_no=12)
ps2ctl.init()
while True:
    key_car= ps2ctl.read_once()   # 收到的字符格式为 keys:UP,RIGHT: pos(lx,ly):0,-1: pos(rx,ry): 0,-1:
    #print("检测到按键:",key_car)    
    key_list= key_car.split(':')  # 用:来将字符串进行分割,写入数组key_list中  key_list[1]输入的按键、key_list[3]左摇杆坐标、key_list[5]右摇杆坐标
    if key_list[1]=="UP": 
      print(key_car," 前进")
      
    if key_list[1]=="UP,LEFT": 
      print(key_car," 左上方")
    time.sleep(0.2)
''' 
  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值