Ethershield v1.1
// Ethersheid Example
// 'Read out temperature fra dallas ds18b20 on a webpage'
// Licence: LGPLv3 open source // dato: 29-11-2016 // Authour: Michael Pedersen, http://techmind.dk
- include "etherShield.h"
// please modify the following two lines. mac and ip have to be unique // in your local area network. You can not have the same numbers in // two devices: static uint8_t mymac[6] = {0x54,0x55,0x58,0x10,0x00,0x24}; static uint8_t myip[4] = {172,16,44,200}; static char baseurl[] = "http://172.16.44.200/"; static uint16_t mywwwport = 80; // listen port for tcp/www (max range 1-254) // or on a different port: //static uint16_t mywwwport = 88; // listen port for tcp/www (max range 1-254)
- define BUFFER_SIZE 500
static uint8_t buf[BUFFER_SIZE+1];
- define STR_BUFFER_SIZE 22
static char strbuf[STR_BUFFER_SIZE+1];
//dallas ds18b20 temperature sensor
- define TEMP_PIN 7
void getCurrentTemp( char *temperature); char sensorData[10]; int i; int wholex;
EtherShield es = EtherShield();
// prepare the webpage by writing the data to the tcp send buffer uint16_t print_webpage (uint8_t *buf); int8_t analyse_cmd (char *str);
void setup() {
Serial.begin(9600); /*initialize enc28j60*/ es.ES_enc28j60Init (mymac); es.ES_enc28j60clkout (2); // change clkout from 6.25MHz to 12.5MHz delay (10);
/* Magjack leds configuration, see enc28j60 datasheet, page 11 */ // LEDA=greed LEDB=yellow
// 0x880 is PHLCON LEDB=on, LEDA=on // enc28j60PhyWrite(PHLCON,0b0000 1000 1000 00 00); es.ES_enc28j60PhyWrite (PHLCON,0x880); delay (500);
// 0x990 is PHLCON LEDB=off, LEDA=off // enc28j60PhyWrite(PHLCON,0b0000 1001 1001 00 00); es.ES_enc28j60PhyWrite (PHLCON,0x990); delay (500);
// 0x880 is PHLCON LEDB=on, LEDA=on // enc28j60PhyWrite(PHLCON,0b0000 1000 1000 00 00); es.ES_enc28j60PhyWrite (PHLCON,0x880); delay (500);
// 0x990 is PHLCON LEDB=off, LEDA=off // enc28j60PhyWrite(PHLCON,0b0000 1001 1001 00 00); es.ES_enc28j60PhyWrite (PHLCON,0x990); delay (500);
// 0x476 is PHLCON LEDA=links status, LEDB=receive/transmit // enc28j60PhyWrite(PHLCON,0b0000 0100 0111 01 10); es.ES_enc28j60PhyWrite (PHLCON,0x476); delay (100);
//init the ethernet/ip layer: es.ES_init_ip_arp_udp_tcp (mymac, myip, mywwwport);
}
void loop() {
Serial.println(wholex); //open the seriel monitor to read out the temperature trough the serial port uint16_t plen, dat_p; int8_t cmd;
plen = es.ES_enc28j60PacketReceive (BUFFER_SIZE, buf);
/*plen will ne unequal to zero if there is a valid packet (without crc error) */ if (plen != 0) {
// arp is broadcast if unknown but a host may also verify the mac // address by sending it to a unicast address. if (es.ES_eth_type_is_arp_and_my_ip (buf,plen)) { es.ES_make_arp_answer_from_request (buf); return; }
// check if ip packets are for us: if (es.ES_eth_type_is_ip_and_my_ip (buf,plen) == 0) { return; }
if (buf[IP_PROTO_P] == IP_PROTO_ICMP_V && buf[ICMP_TYPE_P] == ICMP_TYPE_ECHOREQUEST_V) { es.ES_make_echo_reply_from_request (buf,plen); return; }
// tcp port www start, compare only the lower byte if (buf[IP_PROTO_P] == IP_PROTO_TCP_V&&buf[TCP_DST_PORT_H_P] == 0 && buf[TCP_DST_PORT_L_P] == mywwwport) { if (buf[TCP_FLAGS_P] & TCP_FLAGS_SYN_V) { es.ES_make_tcp_synack_from_syn (buf); // make_tcp_synack_from_syn does already send the syn,ack return; } if (buf[TCP_FLAGS_P] & TCP_FLAGS_ACK_V) { es.ES_init_len_info (buf); // init some data structures dat_p = es.ES_get_tcp_data_pointer(); if (dat_p == 0) { // we can possibly have no data, just ack: if (buf[TCP_FLAGS_P] & TCP_FLAGS_FIN_V) { es.ES_make_tcp_ack_from_any(buf); } return; } if (strncmp ("GET ", (char *) & (buf[dat_p]),4) != 0) { // head, post and other methods for possible status codes see: // http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.htmlplen = es.ES_fill_tcp_data_p (buf, 0, PSTR("HTTP/1.0 200 OK\r\nContent-Type: text/html\r\n\r\n
200 OK
"));goto SENDTCP; } if (strncmp ("/ ", (char *) & (buf[dat_p+4]), 2) == 0) { plen = print_webpage (buf); goto SENDTCP; } cmd = analyse_cmd ((char *) & (buf[dat_p+5])); if (cmd == 1) { plen = print_webpage (buf); }
SENDTCP: es.ES_make_tcp_ack_from_any (buf); // send ack for http get
es.ES_make_tcp_ack_with_data (buf, plen); // send data } } }
}
// The returned value is stored in the global var strbuf
uint8_t find_key_val (char *str,char *key) {
uint8_t found = 0; uint8_t i = 0; char *kp; kp = key; while (*str && *str != ' ' && found == 0) { if (*str == *kp) { kp++; if (*kp == '\0') { str++; kp = key; if (*str == '=') { found = 1; } } } else { kp = key; } str++; } if (found == 1) { // copy the value to a buffer and terminate it with '\0' while (*str && *str!=' ' && *str!='&' && i<STR_BUFFER_SIZE) { strbuf[i] = *str; i++; str++; } strbuf[i] = '\0'; } return (found);
}
int8_t analyse_cmd (char *str)
{
int8_t r=-1;
if (find_key_val(str,"cmd")) { if (*strbuf < 0x3a && *strbuf > 0x2f) { // is a ASCII number, return it r = (*strbuf-0x30); } } return r;
}
uint16_t print_webpage (uint8_t *buf)
{
uint16_t plen; plen = es.ES_fill_tcp_data_p (buf, 0, PSTR("HTTP/1.0 200 OK\r\nContent-Type: text/html\r\n\r\n"));plen = es.ES_fill_tcp_data_p (buf, plen, PSTR("
Welcome to Arduino Ethernet Shield V1.1
")); plen = es.ES_fill_tcp_data_p (buf, plen, PSTR("") );
plen = es.ES_fill_tcp_data_p (buf, plen, PSTR("
Temperature: "));
plen = es.ES_fill_tcp_data_p (buf, plen, PSTR("
"));
i=0;
getCurrentTemp(sensorData); //get the current temperature
while (sensorData[i]) {
buf[TCP_CHECKSUM_L_P+3+plen]=sensorData[i++];
plen++;
}
//plen = es.ES_fill_tcp_data_p (buf, plen, PSTR(sensorData));
plen = es.ES_fill_tcp_data_p (buf, plen, PSTR("
</font>
") );
return (plen);
}
//dallas ds18b20 temperatur sensor function void getCurrentTemp(char *temp) {
int HighByte, LowByte, TReading, Tc_100, sign, whole, fract;
OneWireReset(TEMP_PIN); OneWireOutByte(TEMP_PIN, 0xcc); OneWireOutByte(TEMP_PIN, 0x44); // perform temperature conversion, strong pullup for one sec
OneWireReset(TEMP_PIN); OneWireOutByte(TEMP_PIN, 0xcc); OneWireOutByte(TEMP_PIN, 0xbe);
LowByte = OneWireInByte(TEMP_PIN); HighByte = OneWireInByte(TEMP_PIN); TReading = (HighByte << 8) + LowByte; sign = TReading & 0x8000; // test most sig bit if (sign) // negative { TReading = (TReading ^ 0xffff) + 1; // 2's comp } Tc_100 = (6 * TReading) + TReading / 4; // multiply by (100 * 0.0625) or 6.25
wholex=whole;
whole = Tc_100 / 100; // separate off the whole and fractional portions fract = Tc_100 % 100;
if(sign) temp[2]='-'; else temp[2]='+';
//**database field no 01
temp[0] = 'x'; temp[1] = '1';
//temperature temp[3]= (whole-(whole/100)*100)/10 +'0' ; temp[4]= whole-(whole/10)*10 +'0';
temp[5]='.'; temp[6]=fract/10 +'0'; temp[7]=fract-(fract/10)*10 +'0';
}
void OneWireReset(int Pin) // reset. Should improve to act as a presence pulse {
digitalWrite(Pin, LOW); pinMode(Pin, OUTPUT); // bring low for 500 us delayMicroseconds(500); pinMode(Pin, INPUT); delayMicroseconds(500);
}
void OneWireOutByte(int Pin, byte d) // output byte d (least sig bit first). {
byte n;
for(n=8; n!=0; n--) { if ((d & 0x01) == 1) // test least sig bit { digitalWrite(Pin, LOW); pinMode(Pin, OUTPUT); delayMicroseconds(5); pinMode(Pin, INPUT); delayMicroseconds(60); } else { digitalWrite(Pin, LOW); pinMode(Pin, OUTPUT); delayMicroseconds(60); pinMode(Pin, INPUT); }
d=d>>1; // now the next bit is in the least sig bit position. }
}
byte OneWireInByte(int Pin) // read byte, least sig byte first {
byte d, n, b;
for (n=0; n<8; n++) { digitalWrite(Pin, LOW); pinMode(Pin, OUTPUT); delayMicroseconds(5); pinMode(Pin, INPUT); delayMicroseconds(5); b = digitalRead(Pin); delayMicroseconds(50); d = (d >> 1) | (b<<7); // shift d to right and insert b in most sig bit position } return(d);
} </code>