parent
a31091d88c
commit
c51eacf06c
@ -1,23 +1,15 @@ |
|||||||
# MightyPork's AVR Library |
# "porklib", MightyPork's Arduino Library |
||||||
|
|
||||||
I program my Arduinos in plain C, compile it with `avr-gcc` and flash with `avrdude` (all on Linux). |
This library aims to make development for AVR in C easy and fun. |
||||||
|
|
||||||
Whenever I learn how to do something, I make a library file and put it here. |
Modules of the library provide facilities for pin numbering and aliasing, |
||||||
|
bitwise operations and accessing internal and external peripherals of the AVR. |
||||||
|
|
||||||
The code is tested and optimized for **ATmega328P**, which is used in most Arduinos. I use "Pro Mini" and "Nano". |
It takes some inspiration from Arduino, but is written in pure C, which makes it faster |
||||||
|
and smaller. |
||||||
|
|
||||||
# How to use |
Pull requests to add new modules are welcome, please go ahead! |
||||||
|
|
||||||
Link the `lib/` folder to your project, and make sure you add all lib `.c` files to your `Makefile`, so it builds are the needed code. |
## License |
||||||
|
|
||||||
Some library files don't have `.c`, but many do. |
|
||||||
|
|
||||||
## Useful things |
|
||||||
|
|
||||||
- To easily alias I/O pins, use `lib/pins.h`. |
|
||||||
- For Arduino pins, there are presets in `lib/arduino_pins.h` |
|
||||||
- Binary/byte manipulation utilities are in `lib/calc.h` |
|
||||||
- `lib/meta.h` contains some generally useful things that didn't fit elsewhere |
|
||||||
|
|
||||||
Each header file contains a comment block with explanation, which will help you understand them. |
|
||||||
|
|
||||||
|
The library is provided under MIT license, see the LICENSE file for more info. |
||||||
|
@ -0,0 +1,224 @@ |
|||||||
|
#include <avr/io.h> |
||||||
|
#include <util/delay.h> |
||||||
|
#include <stdint.h> |
||||||
|
#include <stdbool.h> |
||||||
|
|
||||||
|
#include "iopins.h" |
||||||
|
#include "onewire.h" |
||||||
|
|
||||||
|
|
||||||
|
/** Perform bus reset. Returns true if any device is connected */ |
||||||
|
bool ow_reset(const uint8_t pin) |
||||||
|
{ |
||||||
|
as_output_n(pin); |
||||||
|
pin_low_n(pin); |
||||||
|
_delay_us(480); |
||||||
|
|
||||||
|
as_input_pu_n(pin); |
||||||
|
_delay_us(70); |
||||||
|
|
||||||
|
const bool a = get_pin_n(pin); |
||||||
|
|
||||||
|
_delay_us(410); |
||||||
|
|
||||||
|
return a; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** Send a single bit */ |
||||||
|
void ow_tx_bit(const uint8_t pin, const bool bit) |
||||||
|
{ |
||||||
|
as_output_n(pin); |
||||||
|
pin_low_n(pin); |
||||||
|
|
||||||
|
if (bit) { |
||||||
|
_delay_us(6); |
||||||
|
as_input_pu_n(pin); |
||||||
|
_delay_us(64); |
||||||
|
} else { |
||||||
|
_delay_us(60); |
||||||
|
as_input_pu_n(pin); |
||||||
|
_delay_us(10); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** Send a single byte */ |
||||||
|
void ow_send(const uint8_t pin, const uint8_t byte) |
||||||
|
{ |
||||||
|
for (uint8_t i = 0; i < 8; i++) |
||||||
|
{ |
||||||
|
ow_tx_bit(pin, (byte >> i) & 0x01); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** Read a single bit */ |
||||||
|
bool ow_rx_bit(const uint8_t pin) |
||||||
|
{ |
||||||
|
as_output_n(pin); |
||||||
|
pin_low_n(pin); |
||||||
|
_delay_us(6); |
||||||
|
|
||||||
|
as_input_pu_n(pin); |
||||||
|
_delay_us(9); |
||||||
|
|
||||||
|
const bool a = get_pin_n(pin); |
||||||
|
|
||||||
|
_delay_us(55); |
||||||
|
|
||||||
|
return a; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** Read a single byte */ |
||||||
|
uint8_t ow_read(const uint8_t pin) |
||||||
|
{ |
||||||
|
uint8_t byte = 0; |
||||||
|
|
||||||
|
for (uint8_t i = 0; i < 8; i++) |
||||||
|
{ |
||||||
|
byte = (byte >> 1) | (ow_rx_bit(pin) << 7); |
||||||
|
} |
||||||
|
|
||||||
|
return byte; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** Wait until the device is ready. Returns false on timeout */ |
||||||
|
bool ow_wait_ready(const uint8_t pin) |
||||||
|
{ |
||||||
|
uint16_t timeout = 700; |
||||||
|
as_input_pu_n(pin); |
||||||
|
|
||||||
|
while (--timeout > 0) |
||||||
|
{ |
||||||
|
if (is_high_n(pin)) return true; |
||||||
|
_delay_ms(1); |
||||||
|
} |
||||||
|
|
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** Read bytes into an array */ |
||||||
|
void ow_read_arr(const uint8_t pin, uint8_t* array, const uint8_t count) |
||||||
|
{ |
||||||
|
for (uint8_t i = 0; i < count; i++) |
||||||
|
{ |
||||||
|
array[i] = ow_read(pin); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// ---------- CRC utils ----------
|
||||||
|
|
||||||
|
/*
|
||||||
|
Dallas 1-wire CRC routines for Arduino with examples of usage. |
||||||
|
The 16-bit routine is new. |
||||||
|
The 8-bit routine is from http://github.com/paeaetech/paeae/tree/master/Libraries/ds2482/
|
||||||
|
|
||||||
|
Copyright (C) 2010 Kairama Inc |
||||||
|
|
||||||
|
This program is free software: you can redistribute it and/or modify |
||||||
|
it under the terms of the GNU General Public License as published by |
||||||
|
the Free Software Foundation, either version 3 of the License, or |
||||||
|
(at your option) any later version. |
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful, |
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||||
|
GNU General Public License for more details. |
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License |
||||||
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/ |
||||||
|
// Dallas 1-wire 16-bit CRC calculation. Developed from Maxim Application Note 27.
|
||||||
|
|
||||||
|
/** Compute a CRC16 checksum */ |
||||||
|
uint16_t crc16( uint8_t *data, uint8_t len) |
||||||
|
{ |
||||||
|
uint16_t crc = 0; |
||||||
|
|
||||||
|
for (uint8_t i = 0; i < len; i++) |
||||||
|
{ |
||||||
|
uint8_t inbyte = data[i]; |
||||||
|
for (uint8_t j = 0; j < 8; j++) |
||||||
|
{ |
||||||
|
uint8_t mix = (crc ^ inbyte) & 0x01; |
||||||
|
crc = crc >> 1; |
||||||
|
if (mix) |
||||||
|
crc = crc ^ 0xA001; |
||||||
|
|
||||||
|
inbyte = inbyte >> 1; |
||||||
|
} |
||||||
|
} |
||||||
|
return crc; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
// The 1-Wire CRC scheme is described in Maxim Application Note 27:
|
||||||
|
// "Understanding and Using Cyclic Redundancy Checks with Maxim iButton Products"
|
||||||
|
|
||||||
|
/** Compute a CRC8 checksum */ |
||||||
|
uint8_t crc8(uint8_t *addr, uint8_t len) |
||||||
|
{ |
||||||
|
uint8_t crc = 0; |
||||||
|
|
||||||
|
for (uint8_t i = 0; i < len; i++) |
||||||
|
{ |
||||||
|
uint8_t inbyte = addr[i]; |
||||||
|
for (uint8_t j = 0; j < 8; j++) |
||||||
|
{ |
||||||
|
uint8_t mix = (crc ^ inbyte) & 0x01; |
||||||
|
crc >>= 1; |
||||||
|
if (mix) |
||||||
|
crc ^= 0x8C; |
||||||
|
|
||||||
|
inbyte >>= 1; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
return crc; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
// --- utils for DS1820 ---
|
||||||
|
|
||||||
|
/** Read temperature in 0.0625°C, or TEMP_ERROR on error */ |
||||||
|
int16_t ds1820_read_temp(uint8_t pin) |
||||||
|
{ |
||||||
|
ow_send(pin, READ_SCRATCHPAD); |
||||||
|
uint8_t bytes[9]; |
||||||
|
ow_read_arr(pin, bytes, 9); |
||||||
|
|
||||||
|
uint8_t crc = crc8(bytes, 8); |
||||||
|
if (crc != bytes[8]) { |
||||||
|
return TEMP_ERROR; |
||||||
|
} else { |
||||||
|
int16_t a = ((bytes[1] << 8) | bytes[0]) >> 1; |
||||||
|
a = a << 4; |
||||||
|
a += (16 - bytes[6]) & 0x0F; |
||||||
|
a -= 0x04; |
||||||
|
|
||||||
|
return a; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** Read temperature in 0.1°C, or TEMP_ERROR on error */ |
||||||
|
int16_t ds1820_read_temp_c(uint8_t pin) |
||||||
|
{ |
||||||
|
int32_t temp = ds1820_read_temp(pin); |
||||||
|
|
||||||
|
if (temp == TEMP_ERROR) |
||||||
|
return TEMP_ERROR; |
||||||
|
|
||||||
|
temp *= 625; |
||||||
|
uint16_t rem = temp % 1000; |
||||||
|
temp /= 1000; |
||||||
|
if (rem >= 500) temp++; |
||||||
|
|
||||||
|
return (int16_t) temp; |
||||||
|
} |
||||||
|
|
Loading…
Reference in new issue