blob: 267da281edb3f5bdcfb1cecae9cda790d10ae014 [file] [log] [blame]
Damien George78d0dde2015-12-29 00:20:34 +00001/*
2 * This file is part of the MicroPython project, http://micropython.org/
3 *
4 * The MIT License (MIT)
5 *
6 * Copyright (c) 2015 Damien P. George
7 *
8 * Permission is hereby granted, free of charge, to any person obtaining a copy
9 * of this software and associated documentation files (the "Software"), to deal
10 * in the Software without restriction, including without limitation the rights
11 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
12 * copies of the Software, and to permit persons to whom the Software is
13 * furnished to do so, subject to the following conditions:
14 *
15 * The above copyright notice and this permission notice shall be included in
16 * all copies or substantial portions of the Software.
17 *
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
21 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
24 * THE SOFTWARE.
25 */
26
27#include <stdio.h>
28#include <stdint.h>
29
30#include "etshal.h"
31#include "user_interface.h"
32#include "py/obj.h"
33#include "py/mphal.h"
34#include "modpyb.h"
35
36STATIC uint32_t disable_irq(void) {
37 ets_intr_lock();
38 return 0;
39}
40
41STATIC void enable_irq(uint32_t i) {
42 ets_intr_unlock();
43}
44
45STATIC void mp_hal_delay_us_no_irq(uint32_t us) {
46 uint32_t start = system_get_time();
47 while (system_get_time() - start < us) {
48 }
49}
50
51#define DELAY_US mp_hal_delay_us_no_irq
52
53#define TIMING_RESET1 (0)
54#define TIMING_RESET2 (1)
55#define TIMING_RESET3 (2)
56#define TIMING_READ1 (3)
57#define TIMING_READ2 (4)
58#define TIMING_READ3 (5)
59#define TIMING_WRITE1 (6)
60#define TIMING_WRITE2 (7)
61#define TIMING_WRITE3 (8)
62
63static int timings[] = {480, 40, 420, 5, 5, 40, 10, 50, 10};
64
65STATIC mp_obj_t onewire_timings(mp_obj_t timings_in) {
66 mp_obj_t *items;
67 mp_obj_get_array_fixed_n(timings_in, 9, &items);
68 for (int i = 0; i < 9; ++i) {
69 timings[i] = mp_obj_get_int(items[i]);
70 }
71 return mp_const_none;
72}
73STATIC MP_DEFINE_CONST_FUN_OBJ_1(onewire_timings_obj, onewire_timings);
74
75STATIC mp_obj_t onewire_reset(mp_obj_t pin_in) {
76 uint pin = mp_obj_get_pin(pin_in);
77 pin_set(pin, 0);
78 DELAY_US(timings[TIMING_RESET1]);
79 uint32_t i = disable_irq();
80 pin_set(pin, 1);
81 DELAY_US(timings[TIMING_RESET2]);
82 int status = !pin_get(pin);
83 enable_irq(i);
84 DELAY_US(timings[TIMING_RESET3]);
85 return mp_obj_new_bool(status);
86}
87STATIC MP_DEFINE_CONST_FUN_OBJ_1(onewire_reset_obj, onewire_reset);
88
89STATIC int _onewire_readbit(uint pin) {
90 pin_set(pin, 1);
91 uint32_t i = disable_irq();
92 pin_set(pin, 0);
93 DELAY_US(timings[TIMING_READ1]);
94 pin_set(pin, 1);
95 DELAY_US(timings[TIMING_READ2]);
96 int value = pin_get(pin);
97 enable_irq(i);
98 DELAY_US(timings[TIMING_READ3]);
99 return value;
100}
101
102STATIC mp_obj_t onewire_readbit(mp_obj_t pin_in) {
103 return MP_OBJ_NEW_SMALL_INT(_onewire_readbit(mp_obj_get_pin(pin_in)));
104}
105STATIC MP_DEFINE_CONST_FUN_OBJ_1(onewire_readbit_obj, onewire_readbit);
106
107STATIC mp_obj_t onewire_readbyte(mp_obj_t pin_in) {
108 uint pin = mp_obj_get_pin(pin_in);
109 uint8_t value = 0;
110 for (int i = 0; i < 8; ++i) {
111 value |= _onewire_readbit(pin) << i;
112 }
113 return MP_OBJ_NEW_SMALL_INT(value);
114}
115STATIC MP_DEFINE_CONST_FUN_OBJ_1(onewire_readbyte_obj, onewire_readbyte);
116
117STATIC void _onewire_writebit(uint pin, int value) {
118 uint32_t i = disable_irq();
119 pin_set(pin, 0);
120 DELAY_US(timings[TIMING_WRITE1]);
121 pin_set(pin, value);
122 DELAY_US(timings[TIMING_WRITE2]);
123 pin_set(pin, 1);
124 DELAY_US(timings[TIMING_WRITE3]);
125 enable_irq(i);
126}
127
128STATIC mp_obj_t onewire_writebit(mp_obj_t pin_in, mp_obj_t value_in) {
129 _onewire_writebit(mp_obj_get_pin(pin_in), mp_obj_get_int(value_in));
130 return mp_const_none;
131}
132STATIC MP_DEFINE_CONST_FUN_OBJ_2(onewire_writebit_obj, onewire_writebit);
133
134STATIC mp_obj_t onewire_writebyte(mp_obj_t pin_in, mp_obj_t value_in) {
135 uint pin = mp_obj_get_pin(pin_in);
136 int value = mp_obj_get_int(value_in);
137 for (int i = 0; i < 8; ++i) {
138 _onewire_writebit(pin, value & 1);
139 value >>= 1;
140 }
141 return mp_const_none;
142}
143STATIC MP_DEFINE_CONST_FUN_OBJ_2(onewire_writebyte_obj, onewire_writebyte);
144
145STATIC mp_obj_t onewire_crc8(mp_obj_t data) {
146 mp_buffer_info_t bufinfo;
147 mp_get_buffer_raise(data, &bufinfo, MP_BUFFER_READ);
148 uint8_t crc = 0;
149 for (size_t i = 0; i < bufinfo.len; ++i) {
150 uint8_t byte = ((uint8_t*)bufinfo.buf)[i];
151 for (int b = 0; b < 8; ++b) {
152 uint8_t fb_bit = (crc ^ byte) & 0x01;
153 if (fb_bit == 0x01) {
154 crc = crc ^ 0x18;
155 }
156 crc = (crc >> 1) & 0x7f;
157 if (fb_bit == 0x01) {
158 crc = crc | 0x80;
159 }
160 byte = byte >> 1;
161 }
162 }
163 return MP_OBJ_NEW_SMALL_INT(crc);
164}
165STATIC MP_DEFINE_CONST_FUN_OBJ_1(onewire_crc8_obj, onewire_crc8);
166
167STATIC const mp_map_elem_t onewire_module_globals_table[] = {
168 { MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_onewire) },
169
170 { MP_ROM_QSTR(MP_QSTR_timings), MP_ROM_PTR((mp_obj_t)&onewire_timings_obj) },
171 { MP_ROM_QSTR(MP_QSTR_reset), MP_ROM_PTR((mp_obj_t)&onewire_reset_obj) },
172 { MP_ROM_QSTR(MP_QSTR_readbit), MP_ROM_PTR((mp_obj_t)&onewire_readbit_obj) },
173 { MP_ROM_QSTR(MP_QSTR_readbyte), MP_ROM_PTR((mp_obj_t)&onewire_readbyte_obj) },
174 { MP_ROM_QSTR(MP_QSTR_writebit), MP_ROM_PTR((mp_obj_t)&onewire_writebit_obj) },
175 { MP_ROM_QSTR(MP_QSTR_writebyte), MP_ROM_PTR((mp_obj_t)&onewire_writebyte_obj) },
176 { MP_ROM_QSTR(MP_QSTR_crc8), MP_ROM_PTR((mp_obj_t)&onewire_crc8_obj) },
177};
178
179STATIC MP_DEFINE_CONST_DICT(onewire_module_globals, onewire_module_globals_table);
180
181const mp_obj_module_t onewire_module = {
182 .base = { &mp_type_module },
183 .name = MP_QSTR_onewire,
184 .globals = (mp_obj_dict_t*)&onewire_module_globals,
185};