rtt-f030/components/external/espruino/src/jswrap_onewire.c

333 lines
10 KiB
C
Raw Normal View History

2014-02-25 01:47:49 +08:00
/*
* This file is part of Espruino, a JavaScript interpreter for Microcontrollers
*
* Copyright (C) 2013 Gordon Williams <gw@pur3.co.uk>
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* ----------------------------------------------------------------------------
* This file is designed to be parsed during the build process
*
* JavaScript OneWire Functions
* ----------------------------------------------------------------------------
*/
#include "jswrap_onewire.h"
#include "jsdevices.h"
#include "jsinteractive.h"
/*JSON{ "type":"class",
"class" : "OneWire",
"description" : ["This class provides a software-defined OneWire master. It is designed to be similar to Arduino's OneWire library." ]
}*/
static inline Pin onewire_getpin(JsVar *parent) {
return jshGetPinFromVarAndUnLock(jsvObjectGetChild(parent, "pin", 0));
}
/** Reset one-wire, return true if a device was present */
static bool NO_INLINE OneWireReset(Pin pin) {
jshPinSetState(pin, JSHPINSTATE_GPIO_OUT_OPENDRAIN);
//jshInterruptOff();
jshPinSetValue(pin, 0);
jshDelayMicroseconds(500);
jshPinSetValue(pin, 1);
jshDelayMicroseconds(80);
bool hasDevice = !jshPinGetValue(pin);
//jshInterruptOn();
jshDelayMicroseconds(420);
return hasDevice;
}
/** Write 'bits' bits, and return what was read (to read, you must send all 1s) */
static JsVarInt NO_INLINE OneWireRead(Pin pin, int bits) {
jshPinSetState(pin, JSHPINSTATE_GPIO_OUT_OPENDRAIN);
JsVarInt result = 0;
JsVarInt mask = 1;
while (bits-- > 0) {
jshInterruptOff();
jshPinSetValue(pin, 0);
jshDelayMicroseconds(3);
jshPinSetValue(pin, 1);
jshDelayMicroseconds(10); // leave time to let it rise
if (jshPinGetValue(pin))
result = result | mask;
jshInterruptOn();
jshDelayMicroseconds(53);
mask = mask << 1;
}
return result;
}
/** Write 'bits' bits, and return what was read (to read, you must send all 1s) */
static void NO_INLINE OneWireWrite(Pin pin, int bits, JsVarInt data) {
jshPinSetState(pin, JSHPINSTATE_GPIO_OUT_OPENDRAIN);
JsVarInt mask = 1;
while (bits-- > 0) {
if (data & mask) { // short pulse
jshInterruptOff();
jshPinSetValue(pin, 0);
jshDelayMicroseconds(10);
jshPinSetValue(pin, 1);
jshInterruptOn();
jshDelayMicroseconds(55);
} else { // long pulse
jshInterruptOff();
jshPinSetValue(pin, 0);
jshDelayMicroseconds(65);
jshPinSetValue(pin, 1);
jshInterruptOn();
jshDelayMicroseconds(5);
}
mask = mask << 1;
}
}
/*JSON{ "type":"constructor", "class": "OneWire", "name": "OneWire",
"description" : "Create a software OneWire implementation on the given pin",
"generate" : "jswrap_onewire_constructor",
"params" : [ [ "pin", "pin", "The pin to implement OneWire on" ] ],
"return" : [ "JsVar", "A OneWire object" ]
}*/
JsVar *jswrap_onewire_constructor(Pin pin) {
JsVar *ow = jspNewObject(jsiGetParser(), 0, "OneWire");
if (ow) {
JsVar *pinVar = jsvFindChildFromString(ow, "pin", true);
if (pinVar) {
JsVar *pv = jsvNewFromPin(pin);
jsvSetValueOfName(pinVar, pv);
jsvUnLock(pv);
}
jsvUnLock(pinVar);
}
return ow;
}
/*JSON{ "type":"method", "class": "OneWire", "name" : "reset",
"description" : "Perform a reset cycle",
"generate" : "jswrap_onewire_reset",
"return" : [ "bool", "True is a device was present (it held the bus low)" ]
}*/
bool jswrap_onewire_reset(JsVar *parent) {
Pin pin = onewire_getpin(parent);
if (!jshIsPinValid(pin)) return 0;
return OneWireReset(pin);
}
/*JSON{ "type":"method", "class": "OneWire", "name" : "select",
"description" : "Select a ROM - reset needs to be done first",
"generate" : "jswrap_onewire_select",
"params" : [ [ "rom", "int", "The rom to select" ] ]
}*/
void jswrap_onewire_select(JsVar *parent, JsVarInt rom) {
Pin pin = onewire_getpin(parent);
if (!jshIsPinValid(pin)) return;
OneWireWrite(pin, 8, 0x55);
OneWireWrite(pin, 64, rom);
}
/*JSON{ "type":"method", "class": "OneWire", "name" : "skip",
"description" : "Skip a ROM",
"generate" : "jswrap_onewire_skip"
}*/
void jswrap_onewire_skip(JsVar *parent) {
Pin pin = onewire_getpin(parent);
if (!jshIsPinValid(pin)) return;
OneWireWrite(pin, 8, 0xCC);
}
/*JSON{ "type":"method", "class": "OneWire", "name" : "write",
"description" : "Write a byte",
"generate" : "jswrap_onewire_write",
"params" : [ [ "data", "int", "A byte to write" ],
[ "power", "bool", "Whether to leave power on after write (default is false)" ] ]
}*/
void jswrap_onewire_write(JsVar *parent, int data, bool leavePowerOn) {
Pin pin = onewire_getpin(parent);
if (!jshIsPinValid(pin)) return;
OneWireWrite(pin, 8, data);
if (!leavePowerOn) {
jshPinSetState(pin, JSHPINSTATE_GPIO_IN);
jshPinSetValue(pin, 0);
}
}
/*JSON{ "type":"method", "class": "OneWire", "name" : "read",
"description" : "Read a byte",
"generate" : "jswrap_onewire_read",
"return" : [ "int", "The byte that was read" ]
}*/
JsVarInt jswrap_onewire_read(JsVar *parent) {
Pin pin = onewire_getpin(parent);
if (!jshIsPinValid(pin)) return -1;
return OneWireRead(pin, 8);
}
/*JSON{ "type":"method", "class": "OneWire", "name" : "search",
"description" : "Search for devices",
"generate" : "jswrap_onewire_search",
"return" : [ "JsVar", "An array of devices that were found" ]
}*/
JsVar *jswrap_onewire_search(JsVar *parent) {
// search - code from http://www.maximintegrated.com/app-notes/index.mvp/id/187
Pin pin = onewire_getpin(parent);
if (!jshIsPinValid(pin)) return 0;
JsVar *array = jsvNewWithFlags(JSV_ARRAY);
if (!array) return 0;
// global search state
unsigned char ROM_NO[8];
int LastDiscrepancy;
int LastFamilyDiscrepancy;
int LastDeviceFlag;
// reset the search state
LastDiscrepancy = 0;
LastDeviceFlag = FALSE;
LastFamilyDiscrepancy = 0;
int search_result = true;
while (search_result) {
int id_bit_number;
int last_zero, rom_byte_number;
unsigned char id_bit, cmp_id_bit;
unsigned char rom_byte_mask, search_direction;
// initialize for search
id_bit_number = 1;
last_zero = 0;
rom_byte_number = 0;
rom_byte_mask = 1;
search_result = 0;
// if the last call was not the last one
if (!LastDeviceFlag)
{
// 1-Wire reset
if (!OneWireReset(pin))
{
// reset the search
LastDiscrepancy = 0;
LastDeviceFlag = FALSE;
LastFamilyDiscrepancy = 0;
return array;
}
// issue the search command
OneWireWrite(pin, 8, 0xF0);
// loop to do the search
do
{
// read a bit and its complement
id_bit = (unsigned char)OneWireRead(pin, 1);
cmp_id_bit = (unsigned char)OneWireRead(pin, 1);
// check for no devices on 1-wire
if ((id_bit == 1) && (cmp_id_bit == 1))
break;
else
{
// all devices coupled have 0 or 1
if (id_bit != cmp_id_bit)
search_direction = id_bit; // bit write value for search
else
{
// if this discrepancy if before the Last Discrepancy
// on a previous next then pick the same as last time
if (id_bit_number < LastDiscrepancy)
search_direction = ((ROM_NO[rom_byte_number] & rom_byte_mask) > 0);
else
// if equal to last pick 1, if not then pick 0
search_direction = (id_bit_number == LastDiscrepancy);
// if 0 was picked then record its position in LastZero
if (search_direction == 0)
{
last_zero = id_bit_number;
// check for Last discrepancy in family
if (last_zero < 9)
LastFamilyDiscrepancy = last_zero;
}
}
// set or clear the bit in the ROM byte rom_byte_number
// with mask rom_byte_mask
if (search_direction == 1)
ROM_NO[rom_byte_number] |= rom_byte_mask;
else
ROM_NO[rom_byte_number] &= (unsigned char)~rom_byte_mask;
// serial number search direction write bit
OneWireWrite(pin, 1, search_direction);
// increment the byte counter id_bit_number
// and shift the mask rom_byte_mask
id_bit_number++;
rom_byte_mask = (unsigned char)(rom_byte_mask << 1);
// if the mask is 0 then go to new SerialNum byte rom_byte_number and reset mask
if (rom_byte_mask == 0)
{
rom_byte_number++;
rom_byte_mask = 1;
}
}
}
while(rom_byte_number < 8); // loop until through all ROM bytes 0-7
// if the search was successful then
if (!((id_bit_number < 65)))
{
// search successful so set LastDiscrepancy,LastDeviceFlag,search_result
LastDiscrepancy = last_zero;
// check for last device
if (LastDiscrepancy == 0)
LastDeviceFlag = TRUE;
search_result = TRUE;
}
}
// if no device found then reset counters so next 'search' will be like a first
if (!search_result || !ROM_NO[0])
{
LastDiscrepancy = 0;
LastDeviceFlag = FALSE;
LastFamilyDiscrepancy = 0;
search_result = FALSE;
}
if (search_result) {
JsVar *val = jsvNewFromInteger(
(((JsVarInt)ROM_NO[7])<<(7*8)) |
(((JsVarInt)ROM_NO[6])<<(6*8)) |
(((JsVarInt)ROM_NO[5])<<(5*8)) |
(((JsVarInt)ROM_NO[4])<<(4*8)) |
(((JsVarInt)ROM_NO[3])<<(3*8)) |
(((JsVarInt)ROM_NO[2])<<(2*8)) |
(((JsVarInt)ROM_NO[1])<<(1*8)) |
((JsVarInt)ROM_NO[0])
);
if (val) jsvArrayPushAndUnLock(array, val);
}
NOT_USED(LastFamilyDiscrepancy);
}
return array;
}