在mm32f3270上为MicroPython启用Signal模块

本文讲述了如何在MM32F3270嵌入式微控制器上为MicroPython添加Signal模块支持,并揭示了Signal与Pin的绑定机制,重点介绍了`machine_pin.c`和`virpin.c`中的关键代码修改。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在mm32f3270上为MicroPython启用Signal模块

Introduction

原本以为放开在machine的类清单中对Signal类型的定义就能通,虽然能够通过编译,但在实测中并不能操作引脚。

测试MM32F3277中的MicroPython的Signal功能

想想也是,还没搞清楚Signal同Pin的绑定关系,只是心存侥幸就启动测试。实在对不住卓老师还高兴了一晚上。

Signal不像别的组件有很显而易见的代码组织结构。所以还是要从内核开始看代码。

现在追一下代码,看看Signal的实现机制是怎样的。

Tracing

extmod\machine_signal.c

通过不成熟的实验,发现Pin对象还是能够传入到Signal对象的,现在没有绑定到硬件上而已。

on和off函数是通过mp_virtual_pin_write()函数实现的:

STATIC mp_obj_t signal_on(mp_obj_t self_in) {
    mp_virtual_pin_write(self_in, 1);
    return mp_const_none;
}
STATIC MP_DEFINE_CONST_FUN_OBJ_1(signal_on_obj, signal_on);

STATIC mp_obj_t signal_off(mp_obj_t self_in) {
    mp_virtual_pin_write(self_in, 0);
    return mp_const_none;
}
STATIC MP_DEFINE_CONST_FUN_OBJ_1(signal_off_obj, signal_off);

mp_virtual_pin_write() 函数的实现在 extmod/virpin.c 文件

void mp_virtual_pin_write(mp_obj_t pin, int value) {
    mp_obj_base_t *s = (mp_obj_base_t *)MP_OBJ_TO_PTR(pin);
    mp_pin_p_t *pin_p = (mp_pin_p_t *)s->type->protocol;
    pin_p->ioctl(pin, MP_PIN_WRITE, value, NULL);
}

这里调用了mp_pin_p_t中的ioctl函数,它是在virpin.h中声明的,这是哪里定义的呢?

在代码仓库里搜索“mp_pin_p_t”,发现esp系的port中,machine_pin.c中有相关的引用。

STATIC mp_uint_t pin_ioctl(mp_obj_t self_in, mp_uint_t request, uintptr_t arg, int *errcode) {
    (void)errcode;
    machine_pin_obj_t *self = self_in;

    switch (request) {
        case MP_PIN_READ: {
            return gpio_get_level(self->id);
        }
        case MP_PIN_WRITE: {
            gpio_set_level(self->id, arg);
            return 0;
        }
    }
    return -1;
}

STATIC MP_DEFINE_CONST_DICT(machine_pin_locals_dict, machine_pin_locals_dict_table);

STATIC const mp_pin_p_t pin_pin_p = {
    .ioctl = pin_ioctl,
};

const mp_obj_type_t machine_pin_type = {
    { &mp_type_type },
    .name = MP_QSTR_Pin,
    .print = machine_pin_print,
    .make_new = mp_pin_make_new,
    .call = machine_pin_call,
    .protocol = &pin_pin_p,
    .locals_dict = (mp_obj_t)&machine_pin_locals_dict,
};

特别注意,machine_pin_type中的一个字段:“ .protocol = &pin_pin_p,”

但是在mm32的machine_pin_type的实现中就没有这个字段。原来如此 !!!

这也解释了为啥代码能编通,但是执行不对,因为这个protocol的字段是有的,只是默认为空而已。编译不会有问题,运行时不报hardfault就谢天谢地了。

在mm32的machine_pin.c中添加并更新代码如下:

/* to support virpin. */
STATIC mp_uint_t pin_ioctl(mp_obj_t self_in, mp_uint_t request, uintptr_t arg, int *errcode) {
    (void)errcode;
    machine_pin_obj_t *self = self_in;

    switch (request)
    {
        case MP_PIN_READ:
        {
            //return gpio_get_level(self->id);
            return GPIO_ReadInputDataBit(self->gpio_port, 1u << self->gpio_pin) ? 1u: 0u;
        }
        case MP_PIN_WRITE:
        {
            //gpio_set_level(self->id, arg);
            GPIO_WriteBit(self->gpio_port, 1u << self->gpio_pin, arg);
            return 0;
        }
    }
    return -1;
}

STATIC const mp_pin_p_t pin_pin_p =
{
    .ioctl = pin_ioctl,
};

const mp_obj_type_t machine_pin_type =
{
    { &mp_type_type },
    .name        = MP_QSTR_Pin,
    .print       = machine_pin_obj_print, /* __repr__(), which would be called by print(<ClassName>). */
    .call        = machine_pin_obj_call,  /* __call__(), which can be called as <ClassName>(). */
    .make_new    = machine_pin_obj_make_new, /* create new class instance. */
    .protocol    = &pin_pin_p, /* to support virpin. */
    .locals_dict = (mp_obj_dict_t *)&machine_pin_locals_dict,
};

记得要在machine_pin.c中包含extmod/virpin.h

编译通过。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值