Obsah

Microlog2 Source

Info

Verze 2.04

Kompilace projektu

Pozor: Pro kompilaci je potřeba použít starší verzi Arduino IDE, ideálně verzi 1.0.5

V nových verzích se můžete setkat s chybou

microlog2:897: error: 'WebServer_p' was not declared in this scope

nebo

microlog2:899: error: 'WebServer_p' was not declared in this scope

Notes

Download

download

Source

microlog2.ino
// ---[ INFO ]----------------------------------------
 
// Program: MyPower.CZ Microlog2
// License: free for non-commercial use
// Details: info@mypower.cz
 
// ---[ VERSION ]-------------------------------------
 
#define MPWMicroLogVersion     "2.04"
 
// ---[ INCLUDE ]-------------------------------------
 
#include <SPI.h>
#include <EEPROM.h>
#include <Ethernet.h>
#include <utility/w5100.h>
 
// ---[ BUILT-IN CONFIG ]-----------------------------
 
int sendInterval  = 30; // sec default: 120
int sendRetry     = 30; // sec default: 30
byte maxErrors    = 3;  // default: 3
 
byte NETmac[6]    = { 0xF6, 0x88, 0x00, 0x00, 0x00, 0x01 };
//byte NETip[4]     = { 255, 255, 255, 255 }; // = DHCP
byte NETip[4]     = { 192, 168, 1, 188 };  // { 255, 255, 255, 255 }; // = DHCP
byte NETgw[4]     = { 192, 168, 1, 20 };
byte NETmask[4]   = { 255, 255, 255, 0 };
byte NETdns[4]    = { 192, 168, 1, 100 };
 
// ---[ DPIN 22..49 map ]--------------------------
//                    00  01  02  03  04  05  06  07  08  09  10  11  12  13  14  15
//                    [....... LED .....][.not used.][........... RELE .............]
byte DPINmap[16]  = { 40, 42, 44, 46, 48, 0,  0,  0,  22, 23, 24, 25, 26, 27, 28, 29 };
uint16_t DPINstatus    = 0x0000;
//                         <15  08>          <07  >00                          
uint16_t DPINlogicmask = (B00000000 << 8) | B00000011;  // pin/bit  0 = standard logic 0 OFF 1 ON,   1 = negative logic 0 ON 1 OFF
 
// ---[ MODS ]----------------------------------------
 
// mod MyPowerLog
#define modMyPowerLog      1                    // 1=on 0=off
#define modMyPowerHost     "log.mypower.cz"     // log.mypower.cz
#define modMyPowerFVEID    "mojefve1234"        // mojefve1234
 
// mod StausLed
#define modStatusLed       1         // 1=on 0=off
#define modStatusLedPin    41        // default: 40
 
// mod SerialLog
#define modSerialLog       1         // 1=on 0=off
#define modSerialForce     1         
#define modSerialBaudRate  115200    // 9600  ....  115200
 
// mod WebServer
#define modWebServer       1         // 1=on 0=off
#define modWebServerPort   80        // default: 80
 
// mod Display
#define modDisplay         1         // 1=on 0=off
#define modDisplayType     1602      // typ displaye
 
// mod DisplayEmulation
#define modDisplayEmulation     0    // 1=display neni fyzicky pripojen - zobrazeni jen na www rozhrani 
                                     // 0=display pripojen - zobrazeni na fyzickem displayi i na www rozhrani
 
// ---------------------------------------------------
 
byte lxLed=0;
byte lxTime=0;
uint32_t lxSeconds=0;
uint32_t lxSendIndex=0;
byte lxHttpOk=0;
byte lxHttpErrors=0;
 
// ---------------------------------------------------
 
#define dLCDKeyPad1602   1602
 
// ---------------------------------------------------
 
//  TTT GG E   
// 0000 0000
 
#define xpinOPTdefault   B00000000
#define xpinOPTenabled   B00000001
 
#define xpinOPTtypeMask  B01110000
 
#define xpinOPTsample    B00000000
#define xpinOPTavg       B00010000
#define xpinOPTeffective B00100000
#define xpinOPTmin       B00110000
#define xpinOPTmax       B01000000
#define xpinOPTres1      B01010000
#define xpinOPTres2      B01100000
#define xpinOPTdefault   B01110000
 
#define xpinOPTgroupMask B00001100
 
#define xpinOPTsampleG0  B00000000
#define xpinOPTsampleG1  B00000100
#define xpinOPTsampleG2  B00001000
#define xpinOPTsampleG3  B00001100
 
struct tPinConf {
char xname[21];char xunit[6];byte xopt;
signed long xmin,xmax;int xmin10000,xmax10000;
byte chsum;
};
 
struct tDPinConf {
char xname[20];
};
 
#define xComStatusNop               B00000000
#define xComStatusASchanged         B00000001
#define xComStatusDSchanged         B00000010
#define xComStatusIsLastDayInMonth  B00000100
#define xComStatusIsDST             B00001000
 
#define xComStatusXXDaysMonthMask   B00110000
#define xComStatus28DaysMonth       B00000000
#define xComStatus29DaysMonth       B00010000
#define xComStatus30DaysMonth       B00100000
#define xComStatus31DaysMonth       B00110000
 
struct tMpwAccess { 
  byte xComStatus;                    //  flags
  uint16_t mpwremoteDateYear;         //  2013 ...
   uint8_t mpwremoteDateMonth;        //  1-12
   uint8_t mpwremoteDateDay;          //  1-31
   uint8_t mpwremoteDateWeek;         //  1-53
   uint8_t mpwremoteDateDayOfWeek;    //  1 (for Monday) through 7 (for Sunday)
  signed long mpwremoteTimeS;         //  0-86400, -1 - not known
  uint32_t mpwremoteCheckedAtS;       //  local timeS (seconds counter)
  uint32_t mpwlogintimeS;             //  login local timeS 
  uint32_t mpwDeviceId;               //  your device ID on mpw server
  uint32_t mpwAccessId;               //  your access ID on mpw server
} xMpwAccess;
 
struct tMpwDateTime { 
  int xyear, xmonth, xday, xhour, xminute, xsecond; 
  };
 
#define xdcOptBuiltIn      0x00;
#define xdcOptActive       0x01;
 
struct tSampledPin { int pin; byte smpHi8[128]; byte smpLo2[32]; unsigned long smpSqr,smpAvg; int smpMin, smpMax, smpCount; };
struct tPinValue { int smpVal,smpSqr,smpAvg,smpMin,smpMax; unsigned long smpTime; };
struct tSamplerStruct { tPinValue tPinValues[NUM_ANALOG_INPUTS]; tSampledPin tPinSamples[4]; 
  byte tPinTimes[128]; unsigned long tPinMicros; } xSamplerStruct;
 
struct tDeviceConf {
char xfveid[21];
int xsendInterval;
int xsendRetry;
byte xmaxErrors;
byte xdcOpt;
byte chsum;
};
 
struct int64 { word hh,hl,lh,ll; boolean msf; };
 
// ---------------------------------------------------
 
#if modWebServer == 1
EthernetServer server(modWebServerPort);
#endif
 
#if (modDisplay == 1)
byte lcdCurrentPage=0xFF;
byte lcdMaxPage=0;
 
#if (modDisplayType == dLCDKeyPad1602)
struct tDispStats { int xvarscount; int xvarram; };
char _lcdlbuf[17];
char * _lcdweblbuf0=NULL;
char * _lcdweblbuf1=NULL;
tDispStats * _lcdwebdispstats=NULL;
 
#if (modDisplayEmulation == 0)
#include <LiquidCrystal.h>
LiquidCrystal lcd(8, 9, 14, 5, 6, 7);
void lcdClear() { lcd.clear(); }
void lcdInit() { lcd.begin(16, 2); lcdClear(); }
#else
void lcdClear() { }
void lcdInit() { lcdClear(); }
#endif
 
void lcdPrintRight(char * str) { int x=16-strlen(str); lcdPrintAt(str,(x<0?0:x)); }
void lcdPrintCenter(char * str) { int x=8-(strlen(str)/2); lcdPrintAt(str,(x<0?0:x)); }
void lcdPrintLeft(char * str) { lcdPrintAt(str,0); }
void lcdPrintAt(char * str,int x) { 
  int ix=strlen(str);for (int i=0;((i+x<sizeof(_lcdlbuf)-1) && (i<ix));i++) if (x>=0) _lcdlbuf[i+x]=str[i]; }
//  int ix=strlen(str);for (int i=0;((i+x<sizeof(_lcdlbuf)-1) && (i<ix));i++) if ((x+i)>=0) _lcdlbuf[i+x]=str[i]; } ??????
void lcdClearLine() { int i; for (i=0;i<sizeof(_lcdlbuf)-1;i++) _lcdlbuf[i]=32; _lcdlbuf[i]=0; }
void lcdPrintLine(int y) { 
  if ((_lcdweblbuf0!=NULL) && (_lcdweblbuf1!=NULL))
    { char * p=_lcdweblbuf0; if (y==1) p=_lcdweblbuf1;strcpy(p,_lcdlbuf); }
  else { 
#if (modDisplayEmulation == 0) 
  lcd.setCursor(0,y); lcd.print(_lcdlbuf); 
#endif
} }
 
#endif
 
#endif
 
 
// -- [SETUP] ----------------------------------------------
 
void setup()
{
#if modStatusLed == 1
pinMode(modStatusLedPin, OUTPUT); SetLiveLed(HIGH);
#endif
 
 
#if ((modSerialLog == 1) || (modSerialForce == 1))
Serial.begin(modSerialBaudRate);
#endif
 
 
#if ((modSerialLog == 1) || (modDisplay == 1))
char sn1[]="MyPower.CZ";
char sn2[]="MicroLOG";
 
#if (modSerialLog == 1)
XLog(sn1);XLog(sn2);XLog(MPWMicroLogVersion);
 
#if (modMyPowerLog == 1)
xMpwAccess.mpwremoteDateYear=0;
xMpwAccess.mpwremoteDateMonth=0;
xMpwAccess.mpwremoteDateDay=0;
xMpwAccess.mpwremoteDateWeek=0;
xMpwAccess.mpwremoteDateDayOfWeek=0;
xMpwAccess.mpwremoteTimeS=-1;
xMpwAccess.mpwremoteCheckedAtS=0;
xMpwAccess.mpwlogintimeS=0;
xMpwAccess.mpwDeviceId=0;
xMpwAccess.mpwAccessId=0;
xMpwAccess.xComStatus=xComStatusNop;
char cx[50]="+mpwlog @ ";strcat(cx,modMyPowerHost);XLog(cx);
#endif
 
#if (modDisplay == 1)
char ccx[50]="";
#if (modDisplayEmulation == 1 )
sprintf(ccx,"+dispemu @ %04d",modDisplayType);
#else
sprintf(ccx,"+disp @ %04d",modDisplayType);
#endif
XLog(ccx);
#endif
 
#endif
 
#if (modDisplay == 1)
lcdInit();lcdCurrentPage=0xFF;
lcdClearLine();lcdPrintCenter(sn1);lcdPrintLine(0);
lcdClearLine();lcdPrintLeft(sn2);lcdPrintRight(MPWMicroLogVersion);lcdPrintLine(1);
#endif
 
#endif
 
DPINResetHW();
pinMode(53, OUTPUT);digitalWrite(53,HIGH);
 
int xret=0;
 
boolean isdhcp=((NETip[0]&NETip[1]&NETip[2]&NETip[3])==255);
 
if (isdhcp)
  {
#if (modSerialLog == 1)
XLog("DHCP?");
#endif
xret=Ethernet.begin(NETmac);
  }
else
  {
  Ethernet.begin(NETmac, NETip, NETdns, NETgw, NETmask);
  xret=1;
  delay(1000);
  }
 
W5100.setRetransmissionTime(0x07FF);
W5100.setRetransmissionCount(2); 
 
if ( (xret == 0) && (isdhcp) ) {
#if modSerialLog == 1
  XLog("No dhcp.");
#endif
  FlashError(4);
  }
else
  {
#if modSerialLog == 1
  char xtext[100];
  char cf[]="%d.%d.%d.%d";
  IPAddress xaddr;
  XLog("IP,MASK,GW,DNS");
  xaddr=Ethernet.localIP();
  for (byte i=0;i<4;i++) {
    if (i==1) xaddr=Ethernet.subnetMask();else
    if (i==2) xaddr=Ethernet.gatewayIP();else
    if (i==3) xaddr=Ethernet.dnsServerIP();   
    sprintf(xtext,cf,xaddr[0],xaddr[1],xaddr[2],xaddr[3]);
    XLog(xtext);
    }
#endif
  }
 
#if modWebServer == 1
server.begin();
#if modSerialLog == 1
char c[30]="";sprintf(c,"+www @ %d",modWebServerPort);XLog(c);
#endif
#endif
 
SamplerReset();
 
#if modSerialLog == 1
XLog("Done.");
#endif
}
 
 
 
// -- [LOOP] ----------------------------------------------
 
void loop()
{
FlashLiveLed();
CalculateSeconds();
#if modMyPowerLog == 1
SendDataToMyPower();
#endif
#if modWebServer == 1
WebServer();
#endif
#if (modDisplay == 1)
#if (modDisplayType == dLCDKeyPad1602)
DisplayDetectButton();
#endif
#endif
SamplerRun();
}
 
void softReset() { asm volatile ("  jmp 0"); } 
 
#if modSerialLog == 1
void XLog(char * xmsg) { Serial.println(xmsg); }
void XLogRam() { Serial.print("> RAM:"); Serial.println(freeRam()); }
void XLog2(char * xmsg,char * xpre) { Serial.print(xpre);Serial.println(xmsg); }
#endif
 
#if (modDisplay == 1)
void CalcDisplayConfEpromSpace(int * xa,int * la) {
* xa=(sizeof(tPinConf)*NUM_ANALOG_INPUTS)+sizeof(tDeviceConf);
* la=2047; }
#endif
 
 
void SetLiveLed(byte xs) { 
#if modStatusLed==1
digitalWrite(modStatusLedPin,xs);
#endif
}
 
void FlashLiveLed() {
unsigned long i=((millis()/100)%10);
byte qlxLed=( ((i==1)) || ((lxHttpOk==2) && ( (i==3) || (i==5) )) )?HIGH:LOW;
if (qlxLed!=lxLed) { SetLiveLed(qlxLed); lxLed=qlxLed; }
}
 
void CalculateSeconds() {
byte smx=0x7F; int qlxTime=((millis()/1000)&smx);
if (qlxTime!=lxTime) { OneSecondActions();
  lxSeconds+=(lxTime>qlxTime)?(qlxTime+smx+1-lxTime):(qlxTime-lxTime); lxTime=qlxTime; }
}
 
void OneSecondActions() {
#if (modDisplay == 1)
DrawDisplayPage();
#endif
}
 
#if modMyPowerLog==1
void SendDataToMyPower()
{ 
boolean canrun=true; 
for (int xp=0;xp<NUM_ANALOG_INPUTS;xp++)
  if (xSamplerStruct.tPinValues[xp].smpTime==0)
    { canrun=false;break; }
if (canrun) SendDataToMyPower_int(); 
}
 
void SendDataToMyPower_int()
{  
char xtext[200]; 
int xsi=sendInterval;
int xsiorig=xsi;
int xsr=sendRetry;
if (lxHttpOk==2) xsi=xsr;
if (xsi<30) xsi=30;
uint32_t qlxSendIndex=lxSeconds;
if ((lxSendIndex+xsi<=qlxSendIndex) || (lxHttpOk==0))
  {
  lxHttpOk=2;
  lxSendIndex=lxSeconds;
  EthernetClient client;  
  SetLiveLed(HIGH);
#if modSerialLog == 1
  XLogRam();XLog("> Connecting");
#endif
  if ( (client.connect(modMyPowerHost, 80)) && (client.connected()) )
    {
#if modSerialLog == 1
    XLog("> Connected");
#endif
    sprintf(xtext,"GET / HTTP/1.0\r\nHost: %s\r\nUser-Agent: MPWmicrolog/%s\r\nCookie: fveid=%s; ",
      modMyPowerHost,MPWMicroLogVersion,modMyPowerFVEID);
    client.write((byte*)xtext,strlen(xtext));  
 
    boolean islogin=(xMpwAccess.mpwlogintimeS==0);
 
if (islogin)
  {
  strcpy(xtext,"cdate=" __DATE__ " " __TIME__ ";");
  for (int i=0;i<strlen(xtext);i++)
    if (xtext[i]==' ') xtext[i]='_';
  strcat(xtext," ");
  client.write((byte*)xtext,strlen(xtext));
  client.write("login=yes; ");
  }
 
      sprintf(xtext,"utime=%lu/%lu; ",millis(),lxSeconds);
      client.write((byte*)xtext,strlen(xtext));
      sprintf(xtext,"AX=VSAMX; ");
      client.write((byte*)xtext,strlen(xtext));
 
    for (int xp=0;xp<NUM_ANALOG_INPUTS;xp++)
      {
 
      tPinValue * qPinValue;
      qPinValue=&xSamplerStruct.tPinValues[xp];
      sprintf(xtext,"A%d=%X,%X,%X,%X,%X; ",xp,
        qPinValue->smpVal,qPinValue->smpSqr,qPinValue->smpAvg,qPinValue->smpMin,qPinValue->smpMax);
      client.write((byte*)xtext,strlen(xtext));
 
if ( (islogin) || ((xMpwAccess.xComStatus&xComStatusASchanged)!=0) )
  {
      tPinConf pconf;            
      boolean isvalid=ReadPinConf(&pconf,xp);            
      if (isvalid)
        { 
        char smin10k[30]="";sprint10k(smin10k,pconf.xmin,pconf.xmin10000);
        char smax10k[30]="";sprint10k(smax10k,pconf.xmax,pconf.xmax10000);
//        sprintf(xtext,"AC%d=%X/%ld.%04d/%ld.%04d/",xp,pconf.xopt,pconf.xmin,pconf.xmin10000,pconf.xmax,pconf.xmax10000);
        sprintf(xtext,"AC%d=%X/%s/%s/",xp,pconf.xopt,smin10k,smax10k);
        client.write((byte*)xtext,strlen(xtext));
        xtext[0]=0;
        char xf[]="%02X";
        char xh[3];
        byte * p;
        p=(byte*)pconf.xname;
        for (int i=0;i<strlen(pconf.xname);i++) { sprintf(xh,xf,p[i]); strcat(xtext,xh); }
        client.write((byte*)xtext,strlen(xtext));
        strcpy(xtext,"/");
        p=(byte*)pconf.xunit;
        for (int i=0;i<strlen(pconf.xunit);i++) { sprintf(xh,xf,p[i]); strcat(xtext,xh); }
        client.write((byte*)xtext,strlen(xtext));
        client.write("; ");
        }              
  }
      }
 
    xMpwAccess.xComStatus&=~xComStatusASchanged;
 
    client.write("\r\n\r\n");
 
    int lbufpos=0;
    boolean ishead=true;
    xtext[0]=0;
    unsigned long xmstart=millis();
    while (client.connected())
      {
      if (millis()-xmstart>3000) { break; }
      if (client.available()) 
        {
        byte qbuf[200];
        int qlen=client.readBytes((char*)qbuf,sizeof(qbuf)-2);
        for (int i=0;i<qlen;i++) {
          char c = qbuf[i];
          if (c=='\n') {
            if (xtext[0]=='\r') ishead=false; else
            if (!ishead) ParseHTTPResult(xtext);
            lbufpos=0;
            xtext[0]=0;
            }
          else {
            if (lbufpos<sizeof(xtext)-2) {
              xtext[lbufpos]=c; lbufpos++; xtext[lbufpos]=0;
              }
            }
          }
        }
      }
    if (lbufpos>0) ParseHTTPResult(xtext);
    client.stop();
    }
  if (lxHttpOk==1) 
    {
#if modSerialLog == 1
    XLog("> OK");
    sprintf(xtext,"> Waiting (%d seconds) ...",xsiorig);
    XLog(xtext);
#endif
    lxHttpErrors=0;
    }
  else 
    {
    lxHttpErrors++;
#if modSerialLog == 1
    sprintf(xtext,"> ERR (%d/%d)",lxHttpErrors,maxErrors);
    XLog(xtext);
#endif
    if (lxHttpErrors>=maxErrors) FlashError(3);
#if modSerialLog == 1
    else
    sprintf(xtext,"> Retry @ %d seconds ...",xsr);
    XLog(xtext);
#endif
    }
  SetLiveLed(LOW);
  }
}
 
void ParseHTTPResult(char * xtext) { 
#if modSerialLog == 1
XLog2(xtext,"# ");
#endif
if (strcmp(xtext,"result:OK")==0) lxHttpOk=1; else
if (strncmp(xtext,"qid:",4)==0) ParseQID(xtext+4);
}
 
void ParseQID(char * xtext) {
if (xMpwAccess.mpwlogintimeS==0) xMpwAccess.mpwlogintimeS=lxSeconds;
char * p=strchr(xtext,'.');
if (p!=NULL)
  {  
  p[0]=0;
  xMpwAccess.mpwDeviceId=atol(xtext);
  p++;
  char * px=strchr(p,'.');
  if (px!=NULL)
    {
    px[0]=0;
    px++;
    xMpwAccess.mpwAccessId=atol(p);
    if (strlen(px)==20)
      {
      char c[30]="";
      strncpy(c,px,4);c[4]=0;xMpwAccess.mpwremoteDateYear=atoi(c);px+=4;
      strncpy(c,px,2);c[2]=0;xMpwAccess.mpwremoteDateMonth=atoi(c);px+=2;
      strncpy(c,px,2);c[2]=0;xMpwAccess.mpwremoteDateDay=atoi(c);px+=2;
      strncpy(c,px,2);c[2]=0;signed long xhour=atoi(c);px+=2;
      strncpy(c,px,2);c[2]=0;signed long xminute=atoi(c);px+=2;
      strncpy(c,px,2);c[2]=0;signed long xsecond=atoi(c);px+=2;
      strncpy(c,px,2);c[2]=0;byte xdaysinmonth=atoi(c);px+=2;
      strncpy(c,px,1);c[1]=0;byte xisdst=atoi(c);px+=1;
      strncpy(c,px,1);c[1]=0;xMpwAccess.mpwremoteDateDayOfWeek=atoi(c);px+=1;
      strncpy(c,px,2);c[2]=0;xMpwAccess.mpwremoteDateWeek=atoi(c);px+=2;
 
      if (xisdst==1)
        xMpwAccess.xComStatus|=xComStatusIsDST;
      else
        xMpwAccess.xComStatus&=~xComStatusIsDST;
 
      if (xdaysinmonth==xMpwAccess.mpwremoteDateDay)
        xMpwAccess.xComStatus|=xComStatusIsLastDayInMonth;
      else
        xMpwAccess.xComStatus&=~xComStatusIsLastDayInMonth;
 
      xMpwAccess.xComStatus&=~xComStatusXXDaysMonthMask;
      if (xdaysinmonth==28) xMpwAccess.xComStatus|=xComStatus28DaysMonth;
      if (xdaysinmonth==29) xMpwAccess.xComStatus|=xComStatus29DaysMonth;
      if (xdaysinmonth==30) xMpwAccess.xComStatus|=xComStatus30DaysMonth;
      if (xdaysinmonth==31) xMpwAccess.xComStatus|=xComStatus31DaysMonth;
 
      xMpwAccess.mpwremoteTimeS=(xhour*3600)+(xminute*60)+(xsecond);
      xMpwAccess.mpwremoteCheckedAtS=lxSeconds;
      }
    }
 
  }
 
/*  
char xstr[50]="";
tMpwDateTime xMpwDateTime;
GetCurrentDateTime(&xMpwDateTime);
FormatDateTimeStr(xstr, "j.n.Y H:i:s",&xMpwDateTime,sizeof(xstr));
Serial.println(xstr);
*/
}
#endif
 
 
 
void FormatDateTimeStr(char * xstr, char * xfmt, struct tMpwDateTime * xMpwDateTime,int xmaxlen)
{
xstr[0]=0;
while (strlen(xfmt)>0)
  {
  char ch=xfmt[0];
  char cout[20]="";
  if (ch=='d') sprintf(cout,"%02d",xMpwDateTime->xday); else
  if (ch=='j') sprintf(cout,"%d",xMpwDateTime->xday); else
  if (ch=='m') sprintf(cout,"%02d",xMpwDateTime->xmonth); else
  if (ch=='n') sprintf(cout,"%d",xMpwDateTime->xmonth); else
  if (ch=='Y') sprintf(cout,"%d",xMpwDateTime->xyear); else
  if (ch=='y') sprintf(cout,"%d",xMpwDateTime->xyear%99); else
  if (ch=='g') sprintf(cout,"%d",((xMpwDateTime->xhour%12)+1)); else
  if (ch=='G') sprintf(cout,"%d",(xMpwDateTime->xhour)); else
  if (ch=='h') sprintf(cout,"%02d",((xMpwDateTime->xhour%12)+1)); else
  if (ch=='H') sprintf(cout,"%02d",(xMpwDateTime->xhour)); else
  if (ch=='i') sprintf(cout,"%02d",(xMpwDateTime->xminute)); else
  if (ch=='s') sprintf(cout,"%02d",(xMpwDateTime->xsecond)); else
    { cout[0]=ch;cout[1]=0; }
 
  xfmt++;
  if (strlen(xstr)+strlen(cout)<xmaxlen) strcat(xstr,cout); else break;
  }
}
 
int GetCurrentDateTime_DaysInMonth()
{
int xret=28;
int xst=xMpwAccess.xComStatus&xComStatusXXDaysMonthMask;
if (xst==xComStatus28DaysMonth) xret=28; else
if (xst==xComStatus29DaysMonth) xret=29; else
if (xst==xComStatus30DaysMonth) xret=30; else
if (xst==xComStatus31DaysMonth) xret=31;
return xret;
}
 
void GetCurrentDateTime(struct tMpwDateTime * xMpwDateTime)
{
if (xMpwAccess.mpwremoteCheckedAtS>0)
  {
  xMpwDateTime->xyear=xMpwAccess.mpwremoteDateYear;
  xMpwDateTime->xmonth=xMpwAccess.mpwremoteDateMonth;
  xMpwDateTime->xday=xMpwAccess.mpwremoteDateDay;
  uint32_t xtime=lxSeconds-xMpwAccess.mpwremoteCheckedAtS+xMpwAccess.mpwremoteTimeS;
  if (xtime>=86400)
    {
    int xdaysover=xtime/86400UL;
    int xtimeover=xtime%86400UL;
 
    if (xdaysover>25) // max 25 days without time sync
      {
      xMpwDateTime->xyear=0;
      xMpwDateTime->xmonth=0;
      xMpwDateTime->xday=0;
      }
    else
      { 
      int xdaysinmonth=GetCurrentDateTime_DaysInMonth();
      for (int i=0;i<xdaysover;i++)
        {
        if (xMpwDateTime->xday<xdaysinmonth)
          xMpwDateTime->xday++;  
        else
          {
          xMpwDateTime->xday=1;
          if (xMpwDateTime->xmonth<12)
            xMpwDateTime->xmonth++;
          else
            {
            xMpwDateTime->xmonth=1;
            xMpwDateTime->xyear++;
            }
          }
        }
      }
 
    xtime=xtimeover;
    }
  xMpwDateTime->xhour=(xtime/3600)%24;
  int xmod=(xtime%3600);
  xMpwDateTime->xminute=(xmod/60)%60;
  xMpwDateTime->xsecond=xmod%60;
  }
else
  {
  uint32_t xtime=lxSeconds;
  xMpwDateTime->xyear=0;
  xMpwDateTime->xmonth=0;
  xMpwDateTime->xday=(xtime/86400UL);
  xMpwDateTime->xhour=((xtime%86400UL)/3600)%24;
  signed long xmod=((xtime%86400L)%3600);
  xMpwDateTime->xminute=(xmod/60)%60;
  xMpwDateTime->xsecond=xmod%60;    
  }
}
 
 
 
 
 
void FlashError(byte xcount)
{
#if modSerialLog == 1
char xtext[50];
sprintf(xtext,"Error %d.",xcount);
XLog(xtext);
XLog("Flashing err (10x) & restarting.");
#endif
xcount*=2;
for (int i=0;i<10;i++) {
  byte x=0;
  for (int i=0;i<xcount;i++) {
    SetLiveLed((!((x++)&1))?HIGH:LOW);
    delay(250);
    }
  delay(1000);
  }
#if modSerialLog == 1
XLog("Reset");
XLog("---");
#endif
delay(1000);
softReset();
}
 
int freeRam() {
extern unsigned int __heap_start;
extern void *__brkval;
int free_memory; int stack_here;
if (__brkval == 0) free_memory = (int) &stack_here - (int) &__heap_start;
else free_memory = (int) &stack_here - (int) __brkval; 
return (free_memory);
}
 
 
byte calcPchsum(byte xpin, struct tPinConf * pconf) {
byte*xconf=(byte*)pconf; byte xqs=pconf->chsum;
pconf->chsum=0; byte chs=0; for (int i=0;i<sizeof(tPinConf);i++) chs+=(xpin+3)+xconf[i]+i;
pconf->chsum=xqs; return chs&0xFF; }
 
boolean ReadPinConf(struct tPinConf * pconf,int xpin)
{
boolean xret=false;
if (xpin<NUM_ANALOG_INPUTS)
  {
  byte * pcx=(byte*)pconf;
  int sp=sizeof(tPinConf);
//  int xs=(xpin*sp);
  int xs=GetEpromPinAddr(xpin);
  if (xs>=0)
    {
    for (int ax=0;ax<sp;ax++) pcx[ax]=EEPROM.read(ax+xs);
    xret=(calcPchsum(xpin, pconf)==pconf->chsum);
    }
  }
return xret;
}
 
int GetEpromPinAddr(int xpin)
{
int xret=-1;
if (xpin<NUM_ANALOG_INPUTS)
  xret=(xpin*sizeof(tPinConf));
return xret;
}
 
 
// -- [WEB SERVER] ----------------------------------------------
 
 
#if modWebServer == 1
 
#define xWEBhome   0x00
#define xWEBcfgp   0x01
#define xWEBcfg    0x02
#define xWEBmpw    0x03
#define xWEBpins   0x04
 
#define xCFGnone   0x00
#define xCFGk      0x10
#define xCFGv      0x20
#define xCFGs      0x30
 
// w = write
// [w]as = analog settings
// [w]ds = digital settings
// [w]dp = display
// [w]ep = eprom
// hw = hwinfo
// dd = display dump
// db = display button press emulation
 
#define xACTnone   0x00
#define xACThw     0x01
#define xACTas     0x02
#define xACTdp     0x03
#define xACTdd     0x04
#define xACTdb     0x05
#define xACTsp     0x06
#define xACTds     0x07
#define xACTep     0x08
 
#define xACTwds    0x0C
#define xACTwep    0x0D
#define xACTwas    0x0E
#define xACTwdp    0x0F
 
#define xKEYnone          0x00
#define xKEYcmd           0x10
#define xKEYpin           0x20
#define xKEYDispButt      0x30
#define xKEYDpin          0x40
 
#define xPinOptNone       0x00
#define xPinOptEnabled    0x01
#define xPinOptName       0x02
#define xPinOptMin        0x03
#define xPinOptMax        0x04
#define xPinOptUnit       0x05
#define xPinOptType       0x06
#define xPinOptGroup      0x07
#define xPinCmdWrite      0xF0
 
#define xDPinOptNone      0x00
#define xDPinOptName      0x02
 
#define xDPinOptSetStatus 0xA0
 
#define xDPinSetStatus    0xE0
 
void WebServer_pf(char * xstr,signed long * xsl,int * xsl10k) {
int im; *xsl=0;*xsl10k=0; char * p=xstr; char * p10k=p;
for (int i=0;i<strlen(xstr);i++) if (p10k[0]=='.') { p10k[0]=0;p10k++;break; } else p10k++;
boolean hm=(p[0]=='-'); if (hm) { p++; } im=strlen(p);
for (int i=0;i<im;i++) { *xsl*=10; if ((p[i]>='0') && (p[i]<='9')) *xsl+=(p[i]-'0'); }
im=strlen(p10k);
for (int i=0;i<4;i++) { *xsl10k*=10; if ((i<im) && ((p10k[i]>='0') && (p10k[i]<='9'))) 
  *xsl10k+=(p10k[i]-'0'); }
if (hm) *xsl=0-*xsl;
if (hm) *xsl10k=0-*xsl10k; // minus bugfix
}
 
void WebServer_wp(byte xpin, struct tPinConf * pconf) {
if (xpin<NUM_ANALOG_INPUTS) {
  xMpwAccess.xComStatus|=xComStatusASchanged;
  byte*xconf=(byte*)pconf; pconf->chsum=calcPchsum(xpin,pconf);
  int ax=GetEpromPinAddr(xpin);
  if (ax>=0)
    for (int ii=0;ii<sizeof(tPinConf);ii++)
      if (EEPROM.read(ax+ii) != xconf[ii])
        EEPROM.write(ax+ii,xconf[ii]);
  }
}
 
void WebServer_wdp(byte xdpin, struct tDPinConf * pconf) {
if (xdpin<16) {
  xMpwAccess.xComStatus|=xComStatusDSchanged;
  DPIN_epWriteConfName(xdpin,pconf->xname);
  }
}
 
byte WebServer_pb(char * xstr) {
byte cb=0; for (int i=0;i<strlen(xstr);i++) { cb*=10; char c=xstr[i];
  if ((c>='0') && (c<='9')) cb+=c-'0'; }
return cb;
}
 
word WebServer_c(char * lbuf,char c,word xmode, byte * datastruct, unsigned long * xparam) {
byte qmode=xmode&0x0F; byte qsubmode=xmode&0xF0; byte qsaction=(xmode>>8)&0x0F; byte wkey=(xmode>>8)&0xF0;
if (qmode==xWEBhome) {
  if (strncmp(lbuf,"GET ",4)==0) {
    char * p=lbuf+5;
    if (strcmp(p,"cfg")==0) qmode=xWEBcfgp; else
    if (strcmp(p,"pin")==0) qmode=xWEBpins; else
    if (strcmp(p,"mpw")==0) qmode=xWEBmpw;
    if (qmode!=xWEBhome) lbuf[0]=0;
    }
  }
else if (qmode==xWEBcfgp) {
  if (c=='?')
    { qsubmode=xCFGk; lbuf[0]=0; }
  else if ((c=='&') || (c==' '))
    { WebServer_p(lbuf,&qsubmode,&wkey,&qsaction,datastruct,xparam); qsubmode=xCFGk; lbuf[0]=0; } 
  else if (c=='=')
    { WebServer_p(lbuf,&qsubmode,&wkey,&qsaction,datastruct,xparam); 
      if (qsubmode!=xCFGs) { qsubmode=xCFGv; strcpy(lbuf,"\n"); } else  { lbuf[0]=0; }  }
  if (c==' ') { qmode=xWEBcfg; WebServer_e(&qsaction,xparam); } else
  if (qsubmode==xCFGv) WebServer_h(lbuf); else
  if (qsubmode==xCFGs) WebServer_s(lbuf,&qsaction,xparam);
  }
return ((qmode|qsubmode)|((qsaction|wkey)<<8));
}
 
void WebServer_e(byte * qsaction, unsigned long  * xparam)
{
#if modDisplay == 1
if (*qsaction==xACTwdp)
  {
  byte chsum=((*xparam)&0xFF);
  int xaddr=((*xparam>>8)&0xFFFF);
  int xa,la;CalcDisplayConfEpromSpace(&xa,&la);
  int raddr=xaddr+xa+1;
  if (raddr<la) if (EEPROM.read(raddr)!=0) EEPROM.write(raddr,0);   
  if (EEPROM.read(la)!=0) EEPROM.write(la,0);
  if (EEPROM.read(xa)!=chsum) EEPROM.write(xa,chsum);
  }
#endif
}
 
unsigned long WebServer_ph(char * lbuf)
{
unsigned long d=0;
for (byte ii=0;ii<8;ii++) {
  char c=lbuf[ii];
  if (c='\0') break;
  d=(d<<4)+(((c>='0') && (c<='9'))?(c-'0'):(((c>='A') && (c<='F'))?(c-'A'+10):0));
  }
return d;
}
 
void WebServer_s(char * lbuf,byte * qsaction,unsigned long * xparam)
{
#if modDisplay == 1
if (*qsaction==xACTwdp)
  {
  if (strlen(lbuf)==2)
    {
    byte d=0;
    for (byte ii=0;ii<2;ii++) {
      char c=lbuf[ii];
      d=(d<<4)+(((c>='0') && (c<='9'))?(c-'0'):(((c>='A') && (c<='F'))?(c-'A'+10):0));
      }
    lbuf[0]=0;
    if (d==0) d=32;
    byte chsum=((*xparam)&0xFF);
    int xaddr=((*xparam>>8)&0xFFFF);
    int xa,la;CalcDisplayConfEpromSpace(&xa,&la);
    int raddr=xaddr+xa+1;
    if (raddr<la)
      {
      if (EEPROM.read(raddr)!=d) EEPROM.write(raddr,d);
      chsum+=((d&0xFF)+(xaddr+1))&0xFF;     
      xaddr++;
      } 
    *xparam=(chsum|((unsigned long)xaddr<<8));
    }
  }
#endif
}
 
void WebServer_h(char * lbuf)
{
int cx=strlen(lbuf);
char * p=lbuf+cx;
while ((p>=lbuf) && (p[0]!='\n')) p--;
if ((p>=lbuf) && (lbuf+cx-p==3)) {
  byte d=0;
  for (byte ii=1;ii<3;ii++) {
    d=d<<4; char c=p[ii];
    if ((c>='0') && (c<='9')) d+=c-'0';
    if ((c>='A') && (c<='F')) d+=c-'A'+10;
    }
  byte pr='\\';
  if (d==pr) { p[0]=pr;p[1]=pr;p++; } 
  if (d==0) { p[0]=pr;p[1]='0';p++; } 
  else p[0]=d;
  p++;p[0]='\n';p[1]=0;
  }
}
 
void WebServer_p(char * xstr,byte * qsubmode,byte * wkey,byte * qsaction, byte * datastruct, 
  unsigned long * xparam)
{
int ix=strlen(xstr);
if (ix>0){ ix--;xstr[ix]=0; }
if (* qsubmode==xCFGk)
  {
 
#if modDisplay == 1   
  if (*qsaction==xACTdb)
    {
    if (strcmp(xstr,"bid")==0) { *wkey=xKEYDispButt; }
    }    
  else
  if (*qsaction==xACTwdp)
    {
    if (strcmp(xstr,"src")==0) { *qsubmode=xCFGs; *xparam=0; }
    }
  else
#endif
  if (*qsaction==xACTwds)
    {
    char * p=xstr;
    if (strcmp(xstr,"DX")==0)
      {
      byte xpinopt=xDPinSetStatus;
      *xparam&=0xFFFFFF00;
      *xparam|=xpinopt;
      }
    else
    if (p[0]=='s')
      {
      p++;
      int xpinindex=atoi(p)&0x0F;
      byte xpinopt=xDPinOptSetStatus;
      *xparam&=0xFFFF0000;
      *xparam|=xpinopt|(xpinindex<<8);
      }
    else
    if (p[0]=='D')
      {
      p++;
      char * px=p;
      for (int ii=1;ii<strlen(xstr);ii++)
        if (p[0]=='.') { p[0]=0;p++;break; } else p++;
      int xpinindex=atoi(px);
      byte xpinopt=xDPinOptNone;
      if (strcmp(p,"nam")==0) xpinopt=xDPinOptName;
      *xparam&=0xFFFF0000;
      *xparam|=xpinopt|(xpinindex<<8);
      }   
    *wkey=xKEYDpin;
    }
  else
  if (*qsaction==xACTwas)
    {
    char * p=xstr;
    if (p[0]=='A')
      {
      p++;
      char * px=p;
      for (int ii=1;ii<strlen(xstr);ii++)
        if (p[0]=='.') { p[0]=0;p++;break; } else p++;
      byte xpinopt=xPinOptNone;
      if (strcmp(p,"ena")==0) xpinopt=xPinOptEnabled; else
      if (strcmp(p,"nam")==0) xpinopt=xPinOptName; else
      if (strcmp(p,"min")==0) xpinopt=xPinOptMin; else
      if (strcmp(p,"max")==0) xpinopt=xPinOptMax; else
      if (strcmp(p,"uni")==0) xpinopt=xPinOptUnit; else
      if (strcmp(p,"typ")==0) xpinopt=xPinOptType; else
      if (strcmp(p,"grp")==0) xpinopt=xPinOptGroup; else
      if (strcmp(p,"~w")==0) xpinopt=xPinCmdWrite;
      *xparam&=0xFFFFFF00;
      *xparam|=xpinopt;
      }   
    *wkey=xKEYpin;
    }
  else
  if (strcmp(xstr,"cmd")==0) *wkey=xKEYcmd; 
  else
    *wkey=xKEYnone;
  }
else
if (* qsubmode==xCFGv)
  {
  if (ix>0){ ix--;xstr[ix]=0; }
 
#if modDisplay == 1   
  if (*wkey==xKEYDispButt)
    {
    DisplayPressButton(xstr);
    }
  else
#endif
 
// ---------------------------------------
 
  if (*wkey==xKEYDpin) 
    {
    byte xpk=((*xparam)&0xFF);
    tDPinConf * pconf;
    pconf=(tDPinConf*)datastruct;
 
if (xpk==xDPinOptName)
  {
  int xpinindex=(((*xparam)&0x0F00)>>8);
  strncpy(pconf->xname,xstr,sizeof(pconf->xname)-1);
  DPIN_epWriteConfName(xpinindex,pconf->xname);
  for (int ii=0;ii<sizeof(tDPinConf);ii++) datastruct[ii]=0;  
  }
else
if (xpk==xDPinSetStatus) {
  DPINstatus=(uint16_t)WebServer_ph(xstr);
  DPINUpdateHW();  
  }
else
if (xpk==xDPinOptSetStatus) {
  int xpinindex=(((*xparam)&0x0F00)>>8);
  uint16_t xpinstatus=(atoi(xstr))&1;
  DPINstatus&=(~(((uint16_t)1)<<xpinindex));
  DPINstatus|=(xpinstatus<<xpinindex);
  DPINUpdateHW();  
  }
 
 
    }   
else
// ---------------------------------------
  if (*wkey==xKEYpin) 
    {
    byte xpk=((*xparam)&0xFF);
    tPinConf * pconf;
    pconf=(tPinConf*)datastruct;
 
if (xpk==xPinOptEnabled) {
  if (strcmp(xstr,"1")==0)
    pconf->xopt|=xpinOPTenabled;
  }
else
if (xpk==xPinOptName)
  {
  strncpy(pconf->xname,xstr,sizeof(pconf->xname)-1);
  pconf->xname[sizeof(pconf->xname)-1]=0;
  }
else
if (xpk==xPinOptMin) {
  signed long xsl=0;int xsl10000=0;
  WebServer_pf(xstr,&xsl,&xsl10000);
  pconf->xmin=xsl;pconf->xmin10000=xsl10000;
  }
else if (xpk==xPinOptMax) {
  signed long xsl=0;int xsl10000=0;
  WebServer_pf(xstr,&xsl,&xsl10000);
  pconf->xmax=xsl;pconf->xmax10000=xsl10000;
  }
else
if (xpk==xPinOptUnit)
  strncpy(pconf->xunit,xstr,sizeof(pconf->xunit)-1);
else
if (xpk==xPinOptType)
  {
  char w=xstr[0];
  pconf->xopt&=~xpinOPTtypeMask;
  if (w=='s') pconf->xopt|=xpinOPTsample; else
  if (w=='a') pconf->xopt|=xpinOPTavg; else
  if (w=='e') pconf->xopt|=xpinOPTeffective; else
  if (w=='m') pconf->xopt|=xpinOPTmin; else
  if (w=='x') pconf->xopt|=xpinOPTmax;
  }
else
if (xpk==xPinOptGroup)
  {
  char w=xstr[0];
  pconf->xopt&=~xpinOPTgroupMask;
  if (w=='0') pconf->xopt|=xpinOPTsampleG0; else
  if (w=='1') pconf->xopt|=xpinOPTsampleG1; else
  if (w=='2') pconf->xopt|=xpinOPTsampleG2; else
  if (w=='3') pconf->xopt|=xpinOPTsampleG3;
  }
else
if (xpk==xPinCmdWrite) {
    WebServer_wp(WebServer_pb(xstr),(tPinConf*)datastruct);
    for (int ii=0;ii<sizeof(tPinConf);ii++) datastruct[ii]=0;
  }
 
    }   
 // --------------------------------------------
  else
  if (*wkey==xKEYcmd) 
    {
    if (strcmp(xstr,"wep")==0) *qsaction=xACTwep;else
    if (strcmp(xstr,"wds")==0) *qsaction=xACTwds;else
    if (strcmp(xstr,"was")==0) *qsaction=xACTwas;else
    if (strcmp(xstr,"wdp")==0) *qsaction=xACTwdp;else
    if (strcmp(xstr,"hw")==0) *qsaction=xACThw;else
    if (strcmp(xstr,"as")==0) *qsaction=xACTas;else
    if (strcmp(xstr,"ds")==0) *qsaction=xACTds;else
    if (strcmp(xstr,"dp")==0) *qsaction=xACTdp; else
    if (strcmp(xstr,"sp")==0) *qsaction=xACTsp; else
    if (strcmp(xstr,"dd")==0) *qsaction=xACTdd; else
    if (strcmp(xstr,"ep")==0) *qsaction=xACTep; else
    if (strcmp(xstr,"db")==0) *qsaction=xACTdb;    
    }
  }
}
 
void WebServer() {
  EthernetClient client = server.available();
  if (client) {
    boolean cl = true;
    char xln[200]="";
    word xmode=xWEBhome;
    unsigned long xparam=0;
    int cx=sizeof(tPinConf);
    int cxq=sizeof(tDPinConf);if (cx<cxq) cx=cxq;
    byte xDataStruct[cx];
    for (int ii=0;ii<cx;ii++) xDataStruct[ii]=0; 
    int ix=0;
    SetLiveLed(HIGH);
    while (client.connected()) {
      if (client.available()) 
        {
        char c=client.read();
        if (c=='\n') { xln[0]=0; }
        ix=strlen(xln);
        if (ix<sizeof(xln)) { xln[ix]=c;ix++;xln[ix]=0; }
        xmode=WebServer_c(xln,c,xmode,xDataStruct,&xparam);
        byte qmode=xmode&0x0F;
        byte qsubmode=xmode&0xF0; 
        byte qsaction=(xmode>>8)&0x0F; 
        if (c == '\n' && cl) {
          client.write("HTTP/1.1 200 OK\n");
          if (qsaction==xACTep)
            {
            client.write("Content-Disposition: attachment; filename=\"");
            sprintf(xln,"ulog_eprom_%08lX.hex",millis());
            client.write((byte*)xln,strlen(xln));
            client.write("\"\n");
            client.write("Content-Type: text/plain\nConnnection: close\n\n");              
            }
          else
            client.write("Content-Type: text/html\nConnnection: close\n\n");
          if (qmode==xWEBpins) {
            for (int ain = 0; ain < NUM_ANALOG_INPUTS; ain++) {
              if (ain>0) client.write(",");
              sprintf(xln,"A%d:%d",ain,SamplerAnalogRead(ain));
              client.write((byte*)xln,strlen(xln));
              }
            } 
          else
          if (qsaction==xACThw)
            {
            sprintf(xln,"~HW\nt:%lX\nc:%d.",millis(),NUM_ANALOG_INPUTS);
            client.write((byte*)xln,strlen(xln));
            char cq[]="%d.%d.%d.%d.%d.%d.%d.%d";
            sprintf(xln,cq,(int)modStatusLedPin,(int)sendInterval,(int)sendRetry,(int)maxErrors,
            (int)lxHttpErrors,(int)modSerialLog,(int)modDisplay,(modDisplay==1)?(int)modDisplayType:(int)0);
            client.write((byte*)xln,strlen(xln));
            sprintf(xln,".%lu\nf:%s\nh:%s\nx:%d\nr:",(unsigned long)lxSeconds,modMyPowerFVEID,modMyPowerHost,(int)modMyPowerLog);
            client.write((byte*)xln,strlen(xln));
            sprintf(xln,cq,(int)freeRam(),(int)ARDUINO,(int)RAMEND,(int)XRAMEND,
            (int)E2END,(int)FLASHEND,(int)SPM_PAGESIZE,(int)E2PAGESIZE);
            client.write((byte*)xln,strlen(xln));
            sprintf(xln,"\n\nv:%s\nm:%02X%02X%02X%02X%02X%02X\nn:",MPWMicroLogVersion,
            NETmac[0],NETmac[1],NETmac[2],NETmac[3],NETmac[4],NETmac[5],NETmac[6]);
            client.write((byte*)xln,strlen(xln));          
            char cf[]="%d.%d.%d.%d";
            IPAddress xaddr;
            xaddr=Ethernet.localIP();
            for (byte i=0;i<4;i++)
              {
              if (i==1) xaddr=Ethernet.subnetMask();else
              if (i==2) xaddr=Ethernet.gatewayIP();else
              if (i==3) xaddr=Ethernet.dnsServerIP();   
              sprintf(xln,cf,xaddr[0],xaddr[1],xaddr[2],xaddr[3]);
              if (i<3) strcat(xln,"/");
              client.write((byte*)xln,strlen(xln));          
              }
            sprintf(xln,"\nw:%d/%ld\nde:%d",(int)modWebServer,(long)modWebServerPort,(int)modDisplayEmulation);
            client.write((byte*)xln,strlen(xln));
            client.write("\nds:");
            for (int i=0;i<16;i++)
              {
              int qb=DPINmap[i];
              sprintf(xln,"%d",qb);
              if (i<16) strcat(xln,",");
              client.write((byte*)xln,strlen(xln));
              }
            sprintf(xln,"/%04X/%04X\n",DPINstatus,DPINlogicmask);
            client.write((byte*)xln,strlen(xln));          
 
char xstr[30]="";
tMpwDateTime xMpwDateTime;
GetCurrentDateTime(&xMpwDateTime);
FormatDateTimeStr(xstr, "j.n.Y@H:i:s",&xMpwDateTime,sizeof(xstr));
 
 
            sprintf(xln,"mpw:%s/%d.%d.%d.%ld/%lu/%lu/%lu/%lu/%d/%d/%d/%lu\n",xstr,(int)xMpwAccess.mpwremoteDateYear,
              (int)xMpwAccess.mpwremoteDateMonth,
              (int)xMpwAccess.mpwremoteDateDay,
              (signed long)xMpwAccess.mpwremoteTimeS,
              xMpwAccess.mpwremoteCheckedAtS,
              xMpwAccess.mpwlogintimeS,
              xMpwAccess.mpwDeviceId,
              xMpwAccess.mpwAccessId,
              (int)xMpwAccess.xComStatus,
              (int)xMpwAccess.mpwremoteDateWeek,
              (int)xMpwAccess.mpwremoteDateDayOfWeek,
              (unsigned long)(lxSeconds-xMpwAccess.mpwremoteCheckedAtS));
            client.write((byte*)xln,strlen(xln));          
 
            }
          else
// ----------------------------------
          if ((qsaction==xACTds) || (qsaction==xACTwds))
            {
            if (qsaction==xACTwds) client.write("~WDS\n"); else client.write("~DS\n");
            for (int xp=0;xp<16;xp++)
              {
              sprintf(xln,"D%d:%d:",xp,(int)DPINmap[xp]);
              client.write((byte*)xln,strlen(xln));
              char cname[30]="";
              DPIN_epReadConfName(xp,cname,sizeof(cname));
              xln[0]=0;
              char xf[]="%02X";
              char xh[3];
              for (int i=0;i<strlen(cname);i++) {
                int xi=(uint8_t)cname[i];
                sprintf(xh,xf,xi); 
                strcat(xln,xh);
                }
              strcat(xln,"\n");
              client.write((byte*)xln,strlen(xln));
              }
            sprintf(xln,"DX:%04X/%04X\n",DPINstatus,DPINlogicmask);
            client.write((byte*)xln,strlen(xln));
            }
          else
// ----------------------------------
          if ((qsaction==xACTas) || (qsaction==xACTwas))
            {
            if (qsaction==xACTwas) client.write("~WAS\n"); else client.write("~AS\n");
            for (int xp=0;xp<NUM_ANALOG_INPUTS;xp++)
              {
              tPinConf pconf;            
              boolean isvalid=ReadPinConf(&pconf,xp);            
              sprintf(xln,"A%d:",xp);
              client.write((byte*)xln,strlen(xln));
              if (!isvalid)
                strcat(xln,"-");
              else
                { 
                char smin10k[30]="";sprint10k(smin10k,pconf.xmin,pconf.xmin10000);
                char smax10k[30]="";sprint10k(smax10k,pconf.xmax,pconf.xmax10000);
                sprintf(xln,"%X/%s/%s/",pconf.xopt,smin10k,smax10k);
                client.write((byte*)xln,strlen(xln));
                xln[0]=0;
                char xf[]="%02X";
                char xh[3];
                byte * p;
                p=(byte*)pconf.xname;
                for (int i=0;i<strlen(pconf.xname);i++) {
                  sprintf(xh,xf,p[i]); strcat(xln,xh); }
                client.write((byte*)xln,strlen(xln));
                strcpy(xln,"/");
                p=(byte*)pconf.xunit;
                for (int i=0;i<strlen(pconf.xunit);i++) {
                  sprintf(xh,xf,p[i]); strcat(xln,xh); }
                strcat(xln,"/");
                client.write((byte*)xln,strlen(xln));                 
                sprintf(xln,"%d",SamplerAnalogRead(xp));
                }              
              strcat(xln,"\n");
              client.write((byte*)xln,strlen(xln));
              }
            }
          else
          if (qsaction==xACTsp)
            {
            client.write("~SP\n");
            byte qindex=0;
            while (true)
              {
              qindex++;
              byte qpins[4];
              SamplerGet4PinGroup(qpins,&qindex);
              SamplerExec(qpins,&xSamplerStruct,4);
              sprintf(xln,"MS:%lX\nL:%lX\nT:",millis(),xSamplerStruct.tPinMicros);
              client.write((byte*)xln,strlen(xln));
              for (int x=0;x<128;x++)
                {
                sprintf(xln,"%02X",(int)xSamplerStruct.tPinTimes[x]);
                client.write((byte*)xln,strlen(xln));
                }
              client.write("\n");
              for (int p=0;p<4;p++)
                {
                tSampledPin * qSampledPin=&xSamplerStruct.tPinSamples[p];
                int qpin=qSampledPin->pin;
                if (qpin<NUM_ANALOG_INPUTS)
                  {
                  sprintf(xln,"S%d:S=%lu,A=%lu,M=%d,X=%d,C=%d:",qpin,
                    qSampledPin->smpSqr,
                    qSampledPin->smpAvg,
                    qSampledPin->smpMin,
                    qSampledPin->smpMax,
                    qSampledPin->smpCount);
                  client.write((byte*)xln,strlen(xln));
                  for (int x=0;x<128;x++)
                    {
                    sprintf(xln,"%02X",(int)qSampledPin->smpHi8[x]);
                    client.write((byte*)xln,strlen(xln));
                    }
                  client.write("/");
                  for (int x=0;x<32;x++)
                    {
                    sprintf(xln,"%02X",(int)qSampledPin->smpLo2[x]);
                    client.write((byte*)xln,strlen(xln));
                    }
                  client.write("\n");
                  }
                }
              if (qindex==0) break;              
              }
            for (int x=0;x<NUM_ANALOG_INPUTS;x++)
              {
              tPinValue * qPinValue;
              qPinValue=&xSamplerStruct.tPinValues[x];
              sprintf(xln,"A%d:V=%d,S=%d,A=%d,M=%d,X=%d\n",x,
                qPinValue->smpVal,
                qPinValue->smpSqr,
                qPinValue->smpAvg,
                qPinValue->smpMin,
                qPinValue->smpMax);
              client.write((byte*)xln,strlen(xln));
              }
            }
          else
          if ((qsaction==xACTdp) || (qsaction==xACTwdp))
            {
            if (qsaction==xACTwdp)
              client.write("~WDP\n");
            else
              client.write("~DP\n");
#if modDisplay == 1
            sprintf(xln,"T:%d\n",(int)modDisplayType);
            client.write((byte*)xln,strlen(xln));
            int xa,la;
            CalcDisplayConfEpromSpace(&xa,&la);            
            int xs=la-xa;
            sprintf(xln,"B:%d\nD:",xs-1);
            client.write((byte*)xln,strlen(xln));
            byte chsum=0;
            byte rchsum=EEPROM.read(xa);
            for (int i=1;i<xs;i++) 
              {
              int xv=EEPROM.read(xa+i);
              if (xv==0) break;
              chsum+=((xv&0xFF)+i)&0xFF;
              sprintf(xln,"%02X",xv);
              if ((i-1)%10==9) strcat(xln,"\nD:");
              client.write((byte*)xln,strlen(xln));
              }      
            sprintf(xln,"\nR:%02X\nC:%02X",rchsum,chsum);
            client.write((byte*)xln,strlen(xln));
            if (EEPROM.read(la)!=0) EEPROM.write(la,0);
            client.write("\n");
#endif
            }
          else
 
#if ((modDisplay == 1) && (modDisplayType == dLCDKeyPad1602))
          if (qsaction==xACTdd)
            {
            char lcdlbuf0[17]="";
            char lcdlbuf1[17]="";
            _lcdweblbuf0=lcdlbuf0;
            _lcdweblbuf1=lcdlbuf1;
            tDispStats lcdwebdispstats;
            _lcdwebdispstats=&lcdwebdispstats;                     
            DrawDisplayPage();
            client.write("~DD\n");
            sprintf(xln,"S:%d,%d\n",(int)lcdwebdispstats.xvarscount,(int)lcdwebdispstats.xvarram);
            client.write((byte*)xln,strlen(xln));          
            sprintf(xln,"T:%d\nL1:%s\nL2:%s\n",(int)modDisplayType,lcdlbuf0,lcdlbuf1);
            client.write((byte*)xln,strlen(xln));          
            _lcdweblbuf0=NULL;
            _lcdweblbuf1=NULL;
            _lcdwebdispstats=NULL;
            }
          else
#endif
#if (modDisplay == 1)
          if (qsaction==xACTdb)
            {
            client.write("OK");              
            }
          else
#endif         
          if (qsaction==xACTep)
            {
            sprintf(xln,"uLog/%s/eprom.0-%d/@%08lX\n",MPWMicroLogVersion,E2END,millis());
            client.write((byte*)xln,strlen(xln));
            unsigned long xsum=0;
            char s[20]="";
            for (int i=0;i<=E2END;i++)
              {
              if ((i&15)==0) 
                {
                sprintf(xln,"%04X: ",i);
                client.write((byte*)xln,strlen(xln));
                }
              int v=EEPROM.read(i);
              sprintf(xln,"%02X",v);
              int xi=i&15;
              s[xi]=((v>32)&&(v<127))?v:'.';
              s[xi+1]=0;
              client.write((byte*)xln,strlen(xln));          
              xsum=(((xsum>>24)&0xFF)|(xsum<<8))+v+i;
              if ((i&15)==15) 
                {
                sprintf(xln," %s\n",s);
                client.write((byte*)xln,strlen(xln));
                s[0]=0;
                }
              else client.write(" ");
              }
            if (s[0]!=0)
              {
              sprintf(xln," %s\n",s);
              client.write((byte*)xln,strlen(xln));
              s[0]=0;
              }
            sprintf(xln,"S:%08lX",xsum);
            client.write((byte*)xln,strlen(xln));                      
            }
          else
          if (qmode==xWEBcfg)  
            {
            client.write("CFG");
            }
          else
            {
            client.write("<!DOCTYPE HTML>\n<html><head><meta http-equiv=\"Content-Type\" ");
            client.write("content=\"text/html; charset=UTF-8\"><style>");
            client.write("a {color:#FFC000;} body {font-family:sans;font-size:12px;background:#101010;color:white}");
            client.write("</style></head>");
            if (qmode==xWEBhome) client.write("<meta http-equiv=\"refresh\" content=\"5\">");
            client.write("<body>");
            client.write("<b><a href=\"./\">HOME</a> | ");
            client.write("<a href=\"mpw\">TOOLS</a> | ");
            client.write("<a target=\"_blank\" href=\"http://microlog.mypower.cz\">WIKI</a></b><hr>");
            if (qmode==xWEBhome) {
 
 
            for (int x=0;x<NUM_ANALOG_INPUTS;x++)
              {
              tPinValue * qPinValue;
              qPinValue=&xSamplerStruct.tPinValues[x];
              sprintf(xln,"A%d: V=%d, S=%d, A=%d, M=%d, X=%d <br>\n",x,
                qPinValue->smpVal,
                qPinValue->smpSqr,
                qPinValue->smpAvg,
                qPinValue->smpMin,
                qPinValue->smpMax);
              client.write((byte*)xln,strlen(xln));
              }
 
/*              
 
              for (int ain = 0; ain < NUM_ANALOG_INPUTS; ain++) {
                sprintf(xln,"<b>A%d:</b> %d<br>",ain,SamplerAnalogRead(ain));
                client.write((byte*)xln,strlen(xln));
                }
 
*/
              } 
            else
            if (qmode==xWEBmpw) {
              client.write("<script language=\"javascript\" src=\"http://");
              client.write(modMyPowerHost);
              client.write("/microlog-tools?");
              client.write(MPWMicroLogVersion);
              client.write("\"></script>");
              client.write("<div x=\"n\">Connecting to MyPower.CZ ...</div>");
              }
            client.write("<hr><font color=gray size=2><b>powered by ");
            client.write("<a href=\"http://mypower.cz\">MyPower.CZ</a> Microlog</b></font>");
            client.write("</body></html>\n");
            }
          break;
        }
        if (c == '\n') cl = true; else if (c != '\r') cl=false;
        }
      }
    SetLiveLed(LOW);
    delay(1);
    client.stop();
  }
}
#endif
 
 
 
 
// -- [DISPLAY] ----------------------------------------------
 
 
#if (modDisplay == 1)
 
#if (modDisplayType == dLCDKeyPad1602)
 
#define lcdPageSelect   0xFA
#define lcdPageLeft     0xFB
#define lcdPageRight    0xFC
#define lcdPageNone     0xFF
 
struct tDispVar {
char * xname;
signed long xval;
int xval10000;
char * xunit;
};
 
char lcdLastButton=0;
unsigned long lcdLastButtonTime=0;
 
void DisplayDetectButton()
{
char qbutt=0;
int xb=analogRead(0);
if (xb<50) qbutt='R'; else
if (xb<200) qbutt='U'; else
if (xb<400) qbutt='D'; else
if (xb<550) qbutt='L'; else
if (xb<800) qbutt='S';
if ((qbutt!=lcdLastButton) && ((millis()-lcdLastButtonTime)>200))
  {
  if ((lcdLastButton==0) && (qbutt!=0))
    DisplayPressButton(&qbutt);
  lcdLastButton=qbutt;
  lcdLastButtonTime=millis();
  }
 
}
 
void DisplayPressButton(char * xbutt)
{
char p=xbutt[0];
if (p=='X') { delay(1000); softReset(); }
else if (p=='U') { if (lcdCurrentPage==0) lcdCurrentPage=lcdMaxPage; else lcdCurrentPage--; }
else if (p=='D') { if (lcdCurrentPage>=lcdMaxPage) lcdCurrentPage=0; else lcdCurrentPage++; }
else if (p=='R') lcdCurrentPage=lcdPageRight;
else if (p=='L') lcdCurrentPage=lcdPageLeft;
else if (p=='S') lcdCurrentPage=lcdPageSelect;
DrawDisplayPage();
if (lcdCurrentPage>lcdMaxPage) { lcdCurrentPage=0; DrawDisplayPage(); }
}
 
int DisplayGetConfigVarNames(char * xnamebuf,int xbuflen,int xa,int xs)
{
boolean hasvar=false;
char cvn[30]="\n";
strcpy(xnamebuf,cvn);
int xret=0;
for (int i=1;i<=xs;i++)
  {
  char c=EEPROM.read(xa+i);
  if (hasvar)
    {
    if ((c=='.') && ((i<=xs)-2)) { i++;c=EEPROM.read(xa+i);
      if ((c>='0') && (c<='4')) i++;
      c='#';  }
    if ( (DisplayParseVariables_IsVarChar(c)) && (c!='#') )
      {
      int xq=strlen(cvn);
      if (xq<sizeof(cvn)-2) { cvn[xq]=c;xq++;cvn[xq]=0; }
      continue;
      }
    else
      {
      if (cvn[1]!='@')
        {
        strcat(cvn,"\n");
        if (strstr(xnamebuf,cvn)==NULL)
          {
          int qx=(strlen(cvn)-1);
          if (((strlen(xnamebuf)+qx)<(xbuflen-1)) && (qx>1))
            {
            char *p=cvn+1;
            strcat(xnamebuf,p);
            xret++;
            }
          }
        }
      strcpy(cvn,"\n");
      if (c=='#')
        {
        hasvar=false;
        continue;
        }
      }
    }
  if (c==0) break;
  hasvar=(c=='$');
  }
return (xret<1?1:xret);
}
 
struct tDispVar * DisplayFindVarByName(char * xname,byte * xvars, int xmaxvars) {
tDispVar * xret=NULL;
tDispVar * xvar=(tDispVar *)xvars;
for (int i=0;i<xmaxvars;i++) {
  if (strcmp(xvar->xname,xname)==0) { xret=xvar; break; }
  xvar++; }
return xret;
}
 
void DisplayInitConfigVars(char * xnamebuf,byte * xvars, int xmaxvars)
{
char * p=xnamebuf;
if (p[0]=='\n')
  {
  p++;
  tDispVar * xvar=(tDispVar *)xvars;
  char * q=p;
  int ix=strlen(p);
  int qcvars=0;
  for (int i=0;i<ix;i++)
    if (p[i]=='\n') {
      p[i]=0;
      if (q[0]!=0)
        {
        xvar->xname=q;
        xvar->xval=0;
        xvar->xval10000=0;
        xvar->xunit=NULL;    
        qcvars++;
        if (qcvars<xmaxvars) xvar++; else break;
        }
      q=p+1+i;
      }
  }
}
 
void DisplayParseConfig(byte * qWpage)
{
int xa,la;
CalcDisplayConfEpromSpace(&xa,&la);
int xs=la-xa;
byte chsum=0;
byte rchsum=EEPROM.read(xa);
for (int i=1;i<xs;i++) { int xv=EEPROM.read(xa+i);
  if (xv==0) break; chsum+=((xv&0xFF)+i)&0xFF; }      
if (rchsum==chsum) {
  byte xpage=0xFF;
  byte xpageline=0xFF;
  char xln[60]="";
  char xnamebuf[200]="\n";
  int xmaxvars=DisplayGetConfigVarNames(xnamebuf,sizeof(xnamebuf),xa,xs);
  tDispVar xvars[xmaxvars];
  if (_lcdwebdispstats!=NULL) {
    _lcdwebdispstats->xvarscount=xmaxvars;
    _lcdwebdispstats->xvarram=sizeof(xvars);
    }
  DisplayInitConfigVars(xnamebuf,(byte*)xvars, xmaxvars);
  word xflag=lcdPageNone;
  for (int i=1;i<xs;i++) {
    int xv=EEPROM.read(xa+i);
    if ((xv==0) || (xv=='\n')) {
      char * p=xln;
      while ((p[0]!=0) && (p[0]==32)) p++;
      for (int ii=0;p[ii]!=0;ii++) if ((p[ii]=='/') && (p[ii+1]=='/')) { p[ii]=0;break; }
      for (int ii=strlen(p)-1;ii>=0;ii--) if (p[ii]==32) p[ii]=0; else break;
      if (p[0]!=0) DisplayParseConfigL(p,&xpage,&xpageline,qWpage,(byte*)xvars,xmaxvars,&xflag);
      xln[0]=0;
      }
    else 
      { int xls=strlen(xln); if (xls<sizeof(xln)-1) { xln[xls]=xv;xln[xls+1]=0; } }
    if (xv==0) break;
    }
  if (xpage!=0xFF) lcdMaxPage=xpage;
  }
}
 
void DisplayParseConfigL(char * xln,byte * xpage,byte * xpageline,byte * qWpage, byte * xvars, int xmaxvars,word * xflag)
{
if (xln[0]=='#')
  {
  *xflag=lcdPageNone;
  if (xln[1]=='R') *xflag=lcdPageRight; else
  if (xln[1]=='L') *xflag=lcdPageLeft; else
  if (xln[1]=='S') *xflag=lcdPageSelect;
  (*xpage)++;
  (*xpageline)=0xFF;
  }
else
if ((*xpage)==0xFF)
  DisplayParseConfigCmdL(xln, xvars, xmaxvars, xflag);
else
if ((*xpage)!=0xFF)
  {
  if ((*xpageline)==0xFF) (*xpageline)=0; else (*xpageline)++; 
  if (((*qWpage)==(*xpage)) || ((*xflag)==(*qWpage)))
    {
    *qWpage=(*xpage);
    *xflag=lcdPageNone;
    if (*xpageline<2)
      {
      lcdClearLine();
      char * p=xln;
      for (int i=0;i<strlen(xln);i++) if (p[i]=='|') { p+=i;p[0]=0;p++;break; }
      char px[20];
      if (p==xln)
        {       
        DisplayParseVariables(px,xln,sizeof(px),xvars,xmaxvars);
        lcdPrintCenter(px);
        }
      else
        {
        DisplayParseVariables(px,xln,sizeof(px),xvars,xmaxvars);
        lcdPrintLeft(px);
        px[0]=0;
        DisplayParseVariables(px,p,sizeof(px),xvars,xmaxvars);
        lcdPrintRight(px);
        }
      lcdPrintLine(*xpageline);
      }
    }
  }
}
 
#define xVarOpNONE  0x00
#define xVarOpADD   0x01
#define xVarOpSUB   0x02
#define xVarOpMUL   0x03
#define xVarOpDIV   0x04
#define xVarOpEQ    0x05
 
void DisplayParseConfigCmdL(char * xln, byte * xvars, int xmaxvars, word * xflag)
{
byte xop=xVarOpNONE;
tDispVar constVar;
tDispVar * xr=NULL;
tDispVar * xvar=NULL;
int xconstdecs=-1;
boolean hasvar=false;
boolean hasconst=false;
char cvarname[20]="";
for (int i=0;i<=strlen(xln);i++)
  {
  char c=xln[i];
  if ((hasvar) && (!hasconst))
    {
    if (c=='.') { c=xln[i+1]; if ((c>='0') && (c<='9')) { i++; } c='#';  }
    if ( (DisplayParseVariables_IsVarChar(c)) && (c!='#'))
      {
      int xq=strlen(cvarname);
      if (xq<sizeof(cvarname)-1) { cvarname[xq]=c;xq++;cvarname[xq]=0; }
      continue;
      }
    else
      {
      char qunit[10]="";
      constVar.xname=cvarname;
      constVar.xunit=qunit;
      constVar.xval=0;
      constVar.xval10000=0;
      xvar=&constVar;
      if (cvarname[0]=='@')
        {
        char r=cvarname[1];
        char *p=cvarname+2;
        int xpin=0;
        for (int ic=0;ic<strlen(p);ic++)
          { xpin*=10; char e=p[ic]; if ((e>='0') || (e<='9')) xpin+=e-'0'; }
        xvar->xname=cvarname;
        xvar->xunit=qunit;
        xvar->xval=0;
        xvar->xval10000=0;
 
        if ( (xvar->xname[0]=='@') && (xvar->xname[1]=='T') )
          DisplayFillVarByTimeVar(xvar);
        else
          DisplayFillVarByPinValue(xpin,xvar,r);
 
        }
      else
        xvar = DisplayFindVarByName(cvarname,xvars,xmaxvars);
 
      if ((xvar!=NULL) && (xr!=NULL) && (xop!=xVarOpNONE)) {          
        if (xop==xVarOpEQ)
          { xr->xval=xvar->xval; xr->xval10000=xvar->xval10000; }
        else
          DisplayDoVarOp(xr,xvar,xop);
        xop=xVarOpNONE;
        }     
      cvarname[0]=0;
      if (c=='#') { hasvar=false; continue; }
      }
    }
  else
  if ((!hasvar) && (hasconst))
    {
    boolean isnum=((c>='0') && (c<='9'));
    if ((c=='.') && (xconstdecs<0))
      {
      xconstdecs=0;
      continue;
      }
    else
    if ((isnum) && (xvar!=NULL))
      {
      int cn=c-'0';
      if (xconstdecs<0)
        {
        if (xconstdecs>=-9)
          {
          xconstdecs--;
          xvar->xval*=10;
          xvar->xval+=(signed long)cn;
          }
        }
      else
      if (xconstdecs<4)
        {
        xconstdecs++;
        xvar->xval10000*=10;
        xvar->xval10000+=cn;
        }
      continue;
      }
    else
      {
 
      if ((xvar!=NULL) && (xr!=NULL) && (xop!=xVarOpNONE)) {     
        if (xconstdecs>=0) while (xconstdecs<4) { xconstdecs++; xvar->xval10000*=10; }
        if (xop==xVarOpEQ)
          { xr->xval=xvar->xval; xr->xval10000=xvar->xval10000; }
        else
          DisplayDoVarOp(xr,xvar,xop);
        xop=xVarOpNONE;
        }     
 
      xvar=NULL;
      hasconst=false;
      xconstdecs=-1;
      }
    }
 
  if (c==0) break;
  hasvar=(c=='$');
  if (!hasvar) { 
 
    hasconst=((c>='0') && (c<='9'));
    if (hasconst)
      {
      constVar.xval=0;
      constVar.xval10000=0;
      xvar=&constVar;
      i--;
      continue;
      }
 
    if (c!=' ') 
      {      
      byte xlastop=xop;
      if (c=='=') xop=xVarOpEQ; else
      if (c=='+') xop=xVarOpADD; else
      if (c=='-') xop=xVarOpSUB; else
      if (c=='*') xop=xVarOpMUL; else
      if (c=='/') xop=xVarOpDIV;
      if ((xlastop==xVarOpNONE) && (xop==xVarOpEQ) && (xvar!=NULL) && (xr==NULL)) 
        { xr=xvar; xvar=NULL; }
      }
    else
      hasconst=false;
    }
  }
}
 
 
void DisplayDoVarOp(struct tDispVar * xr,struct tDispVar * xvar,byte xop)
{ 
int64 xq1,xq2,xqr;
if (xop==xVarOpMUL) {
  i64x10k2int10kH(&xq1, xr->xval,xr->xval10000);
  i64x10k2int10kH(&xq2, xvar->xval,xvar->xval10000);
} else {
  i64x10k2int10k(&xq1, xr->xval,xr->xval10000);
  i64x10k2int10k(&xq2, xvar->xval,xvar->xval10000);
  }
if (xop==xVarOpADD) i64addS(&xqr,&xq1,&xq2); else
if (xop==xVarOpSUB) i64subS(&xqr,&xq1,&xq2); else
if (xop==xVarOpDIV) { word xmod; i64divW(&xqr, &xmod, &xq1, xvar->xval); } else
if (xop==xVarOpMUL) { int64 xretH; i64mul64(&xretH, &xqr, &xq1, &xq2); }
i64int10k2x10k(&xr->xval,&xr->xval10000,&xqr);
}
 
void PrintComputedPinValue(int xpin,char * xret,int maxunitsize,char xpintype)
{
tDispVar xvar;
DisplayFillVarByPinValue(xpin,&xvar,xpintype);
//signed long xval=xvar.xval;
//int xval10k=xvar.xval10000;
sprint10k(xret,xvar.xval,xvar.xval10000); //minus bugfix
//sprintf(xret,"%ld.%04d",xval,xval10k);
}
 
void DisplayFillVarByPinValue(int xpin,struct tDispVar * xvar,char xpintype)
{
if ((xpin<NUM_ANALOG_INPUTS) && (xvar!=NULL))
  {
  tPinConf pconf;
  byte qpintype=xpinOPTdefault; 
  if ((xpintype=='A') || (xpintype=='a')) qpintype=xpinOPTavg; else
  if ((xpintype=='R') || (xpintype=='r')) qpintype=xpinOPTeffective; else
  if ((xpintype=='S') || (xpintype=='s')) qpintype=xpinOPTsample; else
  if ((xpintype=='M') || (xpintype=='m')) qpintype=xpinOPTmin; else
  if ((xpintype=='X') || (xpintype=='x')) qpintype=xpinOPTmax; 
  int pinval=SamplerAnalogReadType(xpin,qpintype);
  xvar->xval=pinval;
  xvar->xval10000=0;
  if ((xpintype>='A') && (xpintype<='Z'))
    {
    boolean isvalid = ReadPinConf(&pconf,xpin);
    if (isvalid)
      {
      int qxval10000;
      signed long qxval;
      DisplayMapPinValue(pinval,&qxval,&qxval10000,
        pconf.xmin,pconf.xmin10000,pconf.xmax,pconf.xmax10000);
      xvar->xval=qxval;
      xvar->xval10000=qxval10000;
      }
    }
  }
}
 
void DisplayMapPinValue(int pinval,signed long * qxval, int * qxval10k,
      signed long xmin, int xmin10k, signed long xmax, int xmax10k)
{
*qxval10k=0;*qxval=0;int64 qmin,qdif,qx,qml,qmx;i64nul(&qmin);i64nul(&qdif);i64nul(&qx);
i64x10k2int10k(&qmin,xmin,xmin10k);i64x10k2int10k(&qx,xmax,xmax10k);
boolean msf=qmin.msf;i64subS(&qdif,&qx,&qmin);qmin.msf=msf;i64nul(&qx);qx.ll=pinval;
msf=qdif.msf;qdif.msf=false;i64mul64(&qmx,&qml,&qx,&qdif);word xmod;
i64divW(&qmx,&xmod,&qml,1023);qmx.msf=msf;i64addS(&qml,&qmx,&qmin);
i64int10k2x10k(qxval, qxval10k, &qml);
}
 
 
boolean DisplayParseVariables_IsVarChar(char c) {
return ( ((c>='a') && (c<='z')) || ((c>='A') && (c<='Z')) || (c=='_') || 
  (c=='@') || (c=='#') || (c=='.') || ((c>='0') && (c<='9')) ); }
 
void DisplayParseVariables(char * xdst,char * xsrc, int xmaxlen, byte * xvars,int xmaxvars)
{
int ix=0;
boolean hasvar=false;
char cvarname[20]="";
byte xdecs=0xFF;
for (int i=0;i<=strlen(xsrc);i++)
  {
  char c=xsrc[i];
  if (hasvar)
    {
    if (c=='.') { c=xsrc[i+1];
      if (c=='R') { xdecs=0xFE; i++; }  else
      if ((c>='0') && (c<='9')) { xdecs=c-'0';i++; }
      c='#';  }
    if ( (DisplayParseVariables_IsVarChar(c)) && (c!='#'))
      {
      int xq=strlen(cvarname);
      if (xq<sizeof(cvarname)-1) { cvarname[xq]=c;xq++;cvarname[xq]=0; }
      continue;
      }
    else
      {
      char qunit[10]="";
      tDispVar pinVar;
      pinVar.xname=cvarname;
      pinVar.xunit=qunit;
      pinVar.xval=0;
      pinVar.xval10000=0;
      tDispVar * xvar=&pinVar;
      if (cvarname[0]=='@')
        {
        char r=cvarname[1];
        char *p=cvarname+2;
        int xpin=0;
        for (int ic=0;ic<strlen(p);ic++)
          { xpin*=10; char e=p[ic]; if ((e>='0') || (e<='9')) xpin+=e-'0'; }
        xvar->xname=cvarname;
        xvar->xunit=qunit;
        xvar->xval=0;
        xvar->xval10000=0;       
        DisplayFillVarByPinValue(xpin,xvar,r);
        }
      else
        xvar = DisplayFindVarByName(cvarname,xvars,xmaxvars);
 
      if (xvar!=NULL)
        {
        char pxval[50]="";
        if ( (xvar->xname[0]=='@') && (xvar->xname[1]=='T') )
          {
          DisplayGetValueByTimeVar(pxval,xvar->xname,sizeof(pxval));
          }
        else
          {
          char qsc[]=" kMGT";
          char *xsc=qsc;
          signed long xval=xvar->xval;
          int xval10k=xvar->xval10000;
 
          if (xdecs==0xFE)
            {
            sprintf(pxval,"%ld",xval);
            }
          else
            {
            while (xval>1500)
              {
              if (xsc[1]==0) break;
              int64 xq,xr; word xmod;
              i64x10k2int10k(&xq, xval,xval10k);
              i64divW(&xr, &xmod, &xq, 1000);
              i64int10k2x10k(&xval,&xval10k,&xr);
              xsc++;
              }
            sprint10k(pxval,xval,xval10k); // minus bugfix
    //        sprintf(pxval,"%ld.%04d",xval,xval10k);
            DisplayCorrectDecs(pxval,xdecs);
            if ((xsc[0]!=' ') && (xsc[0]!=0)) { xsc[1]=0; strcat(pxval,xsc); }
            }
          }
        for (int cc=0;cc<strlen(pxval);cc++)
          if (ix<xmaxlen-1) { xdst[ix]=pxval[cc];ix++;xdst[ix]=0; }
        }
      cvarname[0]=0;
      if (c=='#') 
        {
        hasvar=false;
        continue;
        }
      }
    }
  if (c==0) break;
  hasvar=(c=='$');
  if ((!hasvar) && (ix<xmaxlen-1)) { xdst[ix]=c;ix++;xdst[ix]=0; }
  }
}
 
 
void DisplayFillVarByTimeVar(struct tDispVar * xvar)
{
char px[30]="";
DisplayGetValueByTimeVar(px,xvar->xname,sizeof(px));
xvar->xval=atol(px);
xvar->xval10000=0;
}
 
 
void DisplayGetValueByTimeVar(char * xbuf,char * xvarname,int xbufmaxsize)
{
xbuf[0]=0;
if ((xvarname[0]=='@') && (xvarname[1]=='T'))
  {
  if (strlen(xvarname)>=3)
    {
char xfmt[5]="";xfmt[0]=xvarname[2];xfmt[1]=0;
tMpwDateTime xMpwDateTime;GetCurrentDateTime(&xMpwDateTime);
FormatDateTimeStr(xbuf,xfmt,&xMpwDateTime,xbufmaxsize);
    }
  }
}
 
void DisplayCorrectDecs(char * pxval, byte xdecs)
{
int xl=strlen(pxval);
if (xl>0) {
  if (xdecs==0xFF) { for (;;) { xl=strlen(pxval); if ((xl==0) || (pxval[xl-1]!='0')) break; pxval[xl-1]=0; } }
  else { if (xdecs>4) xdecs=4; byte qdecs=4-xdecs;
    for (byte xx=0;xx<qdecs;xx++) { xl=strlen(pxval);if (xl==0) break; pxval[xl-1]=0; } }
  xl=strlen(pxval); if (xl>0) if (pxval[xl-1]=='.') pxval[xl-1]=0;
  }
}
 
void DrawDisplayPage()
{
if (lcdCurrentPage==0xFF)
  {
  lcdClear();
  lcdCurrentPage=0;
  }
DisplayParseConfig(&lcdCurrentPage);
}
 
#endif
 
#endif
 
 
 
 
 
 
 
 
 
 
 
 
 
 
// -- [SAMPLER] ----------------------------------------------
 
int SamplerAnalogRead(byte xpin) 
{
return SamplerAnalogReadType(xpin,xpinOPTdefault); 
}
 
 
void SamplerReset() { 
  for (int i=0;i<NUM_ANALOG_INPUTS;i++) 
    xSamplerStruct.tPinValues[i].smpTime=0; 
  }
 
int SamplerAnalogReadType(byte xpin,byte forcextype)
{
int xret=0;
if (xpin<NUM_ANALOG_INPUTS)
  {
  tPinValue * xpinvalue;
  xpinvalue=&xSamplerStruct.tPinValues[xpin];
  byte xtype=SamplerGetPinType(xpin);
  if (forcextype!=xpinOPTdefault) xtype=forcextype;
  if (xtype==xpinOPTavg) xret=xpinvalue->smpAvg;else 
  if (xtype==xpinOPTeffective) xret=xpinvalue->smpSqr;else 
  if (xtype==xpinOPTmin) xret=xpinvalue->smpMin;else 
  if (xtype==xpinOPTmax) xret=xpinvalue->smpMax;else
    xret=xpinvalue->smpVal;
  }
return xret;
}
 
void SamplerRun()
{
byte qpins[4];
SamplerGetNext4PinGroup(qpins); 
SamplerExec(qpins,&xSamplerStruct,4);
}
 
byte SamplerGroupindex=0;
unsigned long Stime=0;
 
void SamplerGetNext4PinGroup(byte * qpins)
{
SamplerGroupindex++;
SamplerGet4PinGroup(qpins,&SamplerGroupindex);
}
 
byte SamplerGetPinType(byte xpin)
{
byte xret=xpinOPTsample;tPinConf pconf;
if (ReadPinConf(&pconf,xpin)) xret=pconf.xopt&xpinOPTtypeMask;
return xret;
}
 
byte SamplerGetPinGroup(byte xpin)
{
byte xret=0;
byte qret=0;
tPinConf pconf;
if (ReadPinConf(&pconf,xpin)) {
  qret=pconf.xopt&xpinOPTgroupMask;
  if (qret==xpinOPTsampleG0) xret=0; else
  if (qret==xpinOPTsampleG1) xret=1; else
  if (qret==xpinOPTsampleG2) xret=2; else
  if (qret==xpinOPTsampleG3) xret=3;
  }
return xret;
}
 
void SamplerGet4PinGroup(byte * qpins,byte * qgroupindex)
{
char pgroups[4][NUM_ANALOG_INPUTS+1];
for (int x=0;x<4;x++) strcpy(pgroups[x],"");
char qch[2]="";qch[1]=0;
for (int x=0;x<NUM_ANALOG_INPUTS;x++) { qch[0]='A'+x; strcat(pgroups[SamplerGetPinGroup(x)],qch); }
char pgroups4[NUM_ANALOG_INPUTS][5];
for (int x=0;x<NUM_ANALOG_INPUTS;x++) strcpy(pgroups4[x],"");
int p4index=0;
for (int x=0;x<4;x++) {
  while (strlen(pgroups[x])>0) {
    strncpy(pgroups4[p4index],pgroups[x],4);
    pgroups4[p4index][4]=0;
    if (strlen(pgroups[x])>4) {
      char * p=(pgroups[x])+4;
      strcpy(pgroups[x],p);
      }
    else
      strcpy(pgroups[x],"");
    p4index++;
    }
  }
for (int p4s=0;p4s<NUM_ANALOG_INPUTS;p4s++) {
  int xl=strlen(pgroups4[p4s]);
  if (xl==4) continue; 
  else {
    int xr=4-xl;
    boolean xfound=false;
    while (!xfound) {
      for (int x=p4s+1;x<NUM_ANALOG_INPUTS;x++)
        if (strlen(pgroups4[x])==xr) {
          strcat(pgroups4[p4s],pgroups4[x]);
          strcpy(pgroups4[x],"");
          xfound=true;
          break;
          }
      if (xfound) break; else if (!xfound) { if (xr>0) xr--; else break; }
      }
    }
  }
byte p4count=0;
for (int x=0;x<NUM_ANALOG_INPUTS;x++) if (strlen(pgroups4[x])==0) { p4count=x;break; }
for (int x=0;x<4;x++) qpins[x]=0xFF;
byte idx=0;
if (p4count>0) {
  idx=*qgroupindex%p4count;
  *qgroupindex=idx;
  char *p=pgroups4[idx];
  int xl=strlen(p);
  if (xl>4) xl=4;
  if (xl!=0)
    for (int x=0;x<xl;x++)
      qpins[x]=(p[x])-'A';
  }
}
 
void SamplerResetPinStruct(int xpin,struct tSampledPin * xSampledPin)
{
xSampledPin->pin=xpin;
for (int x=0;x<sizeof(xSampledPin->smpHi8);x++) xSampledPin->smpHi8[x]=0;
for (int x=0;x<sizeof(xSampledPin->smpLo2);x++) xSampledPin->smpLo2[x]=0;
xSampledPin->smpSqr=0;
xSampledPin->smpAvg=0;
xSampledPin->smpMin=0x3FF;
xSampledPin->smpMax=0x000;
xSampledPin->smpCount=0;
}
 
 
void SamplerExec(byte * pins, struct tSamplerStruct * qSamplerStruct,byte xpinscount)
{
byte * xtimes=qSamplerStruct->tPinTimes;
tSampledPin * xPinSamples=qSamplerStruct->tPinSamples;
unsigned long wqms,wqme;
for (int x=0;x<xpinscount;x++) SamplerResetPinStruct(pins[x],&xPinSamples[x]);
for (int x=0;x<128;x++) xtimes[x]=0;
int qscount=sizeof(xPinSamples[0].smpHi8);
int smpCount=0;
unsigned long xxs,xms,xm1;
unsigned long qms,qm1;
int qsample=0;
int inputpin=0;
tSampledPin * qSampledPin;
int xs3=0;
int xs2=0;
xxs=micros();
xms=0;
qms=0;
while (true) { xm1=micros()-xxs; if ((xm1>>9)!=qms) break; }
xms=xm1;
 
wqms=micros();
for (int x=0;x<qscount;x++)
  {
  xs2=x>>2;
  xs3=(x&0x03)<<1;
  for (int p=0;p<xpinscount;p++)
    {
    qSampledPin=&xPinSamples[p];
    if (qSampledPin->pin<NUM_ANALOG_INPUTS) qsample=analogRead(qSampledPin->pin); else qsample=0;
    qSampledPin->smpHi8[x]=(byte)(qsample>>2);
    qSampledPin->smpLo2[xs2]|=(byte)((qsample&0x03)<<xs3);
    }
  smpCount++;  
  qms=xms>>9;
  while (true) { xm1=micros()-xxs; if ((xm1>>9)!=qms) break; }
  xtimes[x]=(byte)( (xm1-xms) - 400 );
  xms=xm1;
  }
 
wqme=micros()-wqms;
qSamplerStruct->tPinMicros=wqme;
 
for (int x=0;x<qscount;x++)
  {
  xs2=x>>2;
  xs3=(x&0x03)<<1;
  for (int p=0;p<xpinscount;p++)
    {
    qSampledPin=&xPinSamples[p];
    qsample=(((int)qSampledPin->smpHi8[x])<<2)|((qSampledPin->smpLo2[xs2]>>xs3)&0x03);
    qSampledPin->smpAvg+=qsample;
    if (qSampledPin->smpMax<qsample) qSampledPin->smpMax=qsample;
    if (qSampledPin->smpMin>qsample) qSampledPin->smpMin=qsample;
    unsigned long qs=qsample;qs*=qs;
    qSampledPin->smpSqr+=qs;
    }
  }
 
byte qpoint=64;
for (int p=0;p<xpinscount;p++)
  {
  qSampledPin=&xPinSamples[p];
  byte qxpin=qSampledPin->pin;
  if (qxpin<NUM_ANALOG_INPUTS)
    {
    qSampledPin->smpSqr=sqrt(qSampledPin->smpSqr/smpCount);
    qSampledPin->smpAvg=qSampledPin->smpAvg/smpCount;
    qSampledPin->smpCount=smpCount;
    tPinValue * xpinvalue=&qSamplerStruct->tPinValues[qxpin];
    xs3=(qpoint&0x03)<<1;
    xpinvalue->smpVal=(((int)qSampledPin->smpHi8[qpoint])<<2)|((qSampledPin->smpLo2[qpoint>>2]>>xs3)&0x03);;
    xpinvalue->smpSqr=qSampledPin->smpSqr;
    xpinvalue->smpAvg=qSampledPin->smpAvg;
    xpinvalue->smpMin=qSampledPin->smpMin;
    xpinvalue->smpMax=qSampledPin->smpMax;
    xpinvalue->smpTime=millis();
    }
  }
}
 
 
 
 
// -- [MATH10k] -------------------------------------------------
 
boolean isnegative10k(signed long xval, int x10k) { 
  boolean xret=(xval<0);if (xval==0) xret=(x10k<0);return xret; 
}
 
void sprint10k(char * xret, signed long xval, int x10k) {
char xsgn[5]="";if (isnegative10k(xval,x10k)) strcpy(xsgn,"-");
if (xval<0) xval=-xval; if (x10k<0) x10k=-x10k;
sprintf(xret,"%s%ld.%04d",xsgn,xval,x10k);
}
 
// -- [MATH64] -------------------------------------------------
 
void i64print(struct int64 * xval,char * xstr) { 
  char x[200]="";char xm[5]="";if (xval->msf) strcpy(xm,"-");sprintf(x,"--[ %s ] --\n%s%04lX%04lX%04lX%04lX",xstr,xm,
  (long)xval->hh,(long)xval->hl,(long)xval->lh,(long)xval->ll);Serial.println(x); }
 
void i64cpy(struct int64 * xdst,struct int64 * xsrc) { xdst->hh=xsrc->hh;xdst->hl=xsrc->hl;
  xdst->lh=xsrc->lh;xdst->ll=xsrc->ll;xdst->msf=xsrc->msf; }
void i64nul(struct int64 * q1) { q1->hh=0;q1->hl=0;q1->lh=0;q1->ll=0;q1->msf=false; }
void i64shl16(struct int64 * xret) { xret->hh=xret->hl;xret->hl=xret->lh;xret->lh=xret->ll;xret->ll=0;}
int i64cmpS(struct int64 * q1, struct int64 * q2) { for (byte i=0;i<4;i++) {
  word x1=((i==0)?(q1->hh):(i==1?(q1->hl):(i==2?(q1->lh):(i==3?(q1->ll):0))));
  word x2=((i==0)?(q2->hh):(i==1?(q2->hl):(i==2?(q2->lh):(i==3?(q2->ll):0))));
  signed long s1=(signed long)x1; signed long s2=(signed long)x2;
  if (q1->msf) s1=-s1;if (q2->msf) s2=-s2; if (s1<s2) return -1;else if (s1>s2) return 1; }
return 0; }
 
void i64addS(struct int64 * xret, struct int64 * q1, struct int64 * q2) {
i64nul(xret);signed long xx; if ( ((!q1->msf) && (!q2->msf)) || ((q1->msf) && (q2->msf)) ) {
  xx=(signed long)q1->ll+(signed long)q2->ll;xret->ll=(word)(xx&0xFFFF);
  xx=(signed long)q1->lh+(signed long)q2->lh+(signed long)(xx>>16);xret->lh=(word)(xx&0xFFFF);
  xx=(signed long)q1->hl+(signed long)q2->hl+(signed long)(xx>>16);xret->hl=(word)(xx&0xFFFF);
  xx=(signed long)q1->hh+(signed long)q2->hh+(signed long)(xx>>16);xret->hh=(word)(xx&0xFFFF);
  xret->msf=((q1->msf) && (q2->msf)); }
else if ((q1->msf) && (!q2->msf)) { q1->msf=false; i64subS(xret, q2, q1); }
else if ((!q1->msf) && (q2->msf)) { q2->msf=false; i64subS(xret, q1, q2); }
}
 
void i64subS(struct int64 * xret, struct int64 * q1, struct int64 * q2) { 
i64nul(xret); if ((!q1->msf) && (q2->msf)) { q2->msf=false; i64addS(xret, q1, q2); } else
if ((q1->msf) && (!q2->msf)) { q1->msf=false; i64addS(xret, q1, q2); xret->msf=true; } else
if ((q1->msf) && (q2->msf)) { q1->msf=false;q2->msf=false; i64subS(xret, q2, q1); } else 
if ((!q1->msf) && (!q2->msf)) { int xcmp=i64cmpS(q1,q2);
  if (xcmp==0) return; else if (xcmp==-1) { i64subS(xret, q2, q1); xret->msf=true; }
  else { signed long xx,xh,xl;xx=(signed long)q1->ll-(signed long)q2->ll;
    xh=(xx>=0)?0:1;xret->ll=(word)xx;xx=(signed long)q1->lh-(signed long)q2->lh-xh;
    xh=(xx>=0)?0:1;xret->lh=(word)xx;xx=(signed long)q1->hl-(signed long)q2->hl-xh;
    xh=(xx>=0)?0:1;xret->hl=(word)xx;xx=(signed long)q1->hh-(signed long)q2->hh-xh;
    xret->hh=(word)xx; } }
}
 
boolean i64divW(struct int64 * xret, word * xmod, struct int64 * q1, word q2) {
i64nul(xret);*xmod=0; if (q2==0) return false;
else { unsigned long ql;ql=(unsigned long)q1->hh;xret->ll=ql/q2;*xmod=ql%q2;i64shl16(xret);
  ql=(((unsigned long)(*xmod))<<16)+(unsigned long)q1->hl;xret->ll=ql/q2;*xmod=ql%q2;i64shl16(xret);
  ql=(((unsigned long)(*xmod))<<16)+(unsigned long)q1->lh;xret->ll=ql/q2;*xmod=ql%q2;i64shl16(xret);
  ql=(((unsigned long)(*xmod))<<16)+(unsigned long)q1->ll;xret->ll=ql/q2;*xmod=ql%q2;
  return true; } 
}
 
unsigned long i64lo(struct int64 * xint) { return (((unsigned long)xint->lh)<<16)+((unsigned long)xint->ll);}
unsigned long i64hi(struct int64 * xint) { return (((unsigned long)xint->hh)<<16)+((unsigned long)xint->hl);}
 
void i64mul64(struct int64 * xretH, struct int64 * xretL, struct int64 * q1, struct int64 * q2) 
{ i64nul(xretH);i64nul(xretL);unsigned long q1l,q1h,q2l,q2h;
q1l=i64lo(q1);q1h=i64hi(q1);q2l=i64lo(q2);q2h=i64hi(q2);int64 xr1,xr2,xr3,xr4;
i64mul(&xr4, q1h, q2h);i64mul(&xr3, q1h, q2l);i64mul(&xr2, q1l, q2h); i64mul(&xr1, q1l, q2l); 
xretL->ll=xr1.ll;xretL->lh=xr1.lh;unsigned long xr2h=i64hi(&xr2);unsigned long xr3h=i64hi(&xr3);
xr2.hh=0;xr2.hl=0;xr3.hh=0;xr3.hl=0;int64 xr1x;i64nul(&xr1x);
xr1x.lh=xr1.hh;xr1x.ll=xr1.hl;int64 xr23sumx;i64addS(&xr23sumx,&xr2,&xr3);int64 xr23sumL;
i64addS(&xr23sumL,&xr23sumx,&xr1x);xretL->hl=xr23sumL.ll;xretL->hh=xr23sumL.lh;
int64 xr23o;i64nul(&xr23o);xr2.lh=(word)(xr2h>>16);xr2.ll=(word)(xr2h&0xFFFF);
xr3.lh=(word)(xr3h>>16);xr3.ll=(word)(xr3h&0xFFFF);xr23o.ll=xr23sumL.hl;xr23o.lh=xr23sumL.hh;
i64addS(&xr1,&xr4,&xr2);i64addS(&xr4,&xr1,&xr3);i64addS(xretH,&xr4,&xr23o);
}
 
void i64mul(struct int64 * xret, unsigned long q1, unsigned long q2) { 
unsigned long x1,x2,qp,qp1;i64nul(xret);
word all=(word)q1;word alh=(word)(q1>>16);word bll=(word)q2;word blh=(word)(q2>>16);
x1=(unsigned long)all*(unsigned long)bll;xret->ll=(word)(x1);qp=(x1>>16);
x1=(unsigned long)alh*(unsigned long)bll;qp1=(x1>>16);x1&=0xFFFF;
x2=(unsigned long)all*(unsigned long)blh;qp1+=(x2>>16);x2&=0xFFFF;x1+=x2+qp;qp=(x1>>16)+qp1;
xret->lh=(word)(x1);x1=((unsigned long)alh*(unsigned long)blh);
x1+=qp;xret->hl=(word)(x1);xret->hh=(word)(x1>>16);
}
 
void i64int10k2x10k(signed long * xval, int * x10k, struct int64 * xint64) {
boolean msf=xint64->msf;xint64->msf=false;int64 qret;i64nul(&qret);
word xmod=0;i64divW(&qret, &xmod, xint64, 10000);*x10k=xmod;
signed long qval=(((unsigned long)qret.lh)<<16)+(unsigned long)qret.ll;
*xval=msf?(-qval):qval; if (msf) *x10k=-(*x10k); // minus bugfix
xint64->msf=msf;
}
 
void i64x10k2int10k(struct int64 * xret, signed long xval, int x10k) {
int64 q1;int64 q2;i64nul(xret);i64nul(&q1);i64nul(&q2);
/* boolean msf=((xval)<0); */ boolean msf=isnegative10k(xval,x10k); // minus bugfix
if (msf) xval=-(xval);if (x10k<0) x10k=-(x10k);
q2.ll=(x10k)&0xFFFF;i64mul(&q1, (unsigned long)(xval), 10000UL);
i64addS(xret, &q1, &q2);xret->msf=msf;
}
 
void i64x10k2int10kH(struct int64 * xret, signed long xval, int x10k) {
int64 q1;int64 q2;i64nul(xret);i64nul(&q1);i64nul(&q2);
/* boolean msf=((xval)<0); */ boolean msf=isnegative10k(xval,x10k); // minus bugfix
if (msf) xval=-(xval);if (x10k<0) x10k=-(x10k);
q2.ll=(x10k/100)&0xFFFF;i64mul(&q1, (unsigned long)(xval), 100UL);
i64addS(xret, &q1, &q2);xret->msf=msf;
}
 
 
// ---[ Digital Pin Output ]----------------------------------------
 
int DPIN_epGetConfNameAddr(byte xpinindex) { 
return (xpinindex<16)?(2048+(xpinindex*20)):-1;
}
 
void DPIN_epWriteConfName(byte xpinindex,char * xnamebuf ) {
int xaddr=DPIN_epGetConfNameAddr(xpinindex);
if (xaddr>0) 
  {
  byte chsum=(xaddr&0xFF)+xpinindex;
  for (int i=1;i<20;i++) {
    byte ch=xnamebuf[i-1]; 
    chsum+=ch+i+(ch&(0x0F<<2));
    if (EEPROM.read(xaddr+i)!=ch) EEPROM.write(xaddr+i,ch);    
    if (ch==0) break;
    }
  if (EEPROM.read(xaddr)!=chsum) EEPROM.write(xaddr,chsum);    
  }
}
 
void DPIN_epReadConfName(byte xpinindex,char * xnamebuf, int xbufsize ) {
char xbuf[20]="";
int xaddr=DPIN_epGetConfNameAddr(xpinindex);
byte chsum=(xaddr&0xFF)+xpinindex;
if (xaddr>0) 
  {
  for (int i=1;i<20;i++) {
    byte ch=EEPROM.read(xaddr+i);
    chsum+=ch+i+(ch&(0x0F<<2));
    if (ch==0) break;
    xbuf[i-1]=ch;
    xbuf[i]=0;
    }
  if (EEPROM.read(xaddr)==chsum)
    {
    strncpy(xnamebuf,xbuf,xbufsize-1);
    xnamebuf[xbufsize-1]=0;
    }
  else
    xnamebuf[0]=0;
  }
}
 
void DPINUpdateHW() {
uint16_t dstat=DPINstatus^DPINlogicmask;
for (int i=0;i<16;i++) { byte xdpin=DPINmap[i];
  if ((xdpin>=22) && (xdpin<=49)) digitalWrite(xdpin, ((dstat&(1<<i))==0)?LOW:HIGH);
  }
}
 
void DPINResetHW() {
for (int i=0;i<16;i++) { byte xdpin=DPINmap[i];
  if ((xdpin>=22) && (xdpin<=49)) { pinMode(xdpin, OUTPUT); digitalWrite(xdpin, LOW); }
  }
DPINUpdateHW();
}