#include #include #include #include "basic/basic.h" #include "funk/mesh.h" #include "funk/nrf24l01p.h" #include "basic/byteorder.h" #include "basic/random.h" #include "basic/config.h" #include "lcd/print.h" char meshgen=0; // Generation char meshincctr=0; char meshmsg=0; char meshnice=0; MPKT meshbuffer[MESHBUFSIZE]; #include "SECRETS" struct NRF_CFG oldconfig; static int mesh_gt(char curgen, char newgen){ unsigned char dif=curgen-newgen; if(curgen==0) return 1; return (dif>128); }; void initMesh(void){ for(int i=0;i0x7f || MO_TYPE(pkt)<0x20) return MP_SEND|MP_RECV; if(MO_TYPE(pkt)=='T' && MO_BODY(pkt)[5]) return MP_SEND|MP_RECV; if(MO_TYPE(pkt)=='T' && MO_TIME(pkt)<86400) return MP_OK; if(MO_TYPE(pkt)>='A' && MO_TYPE(pkt)<='Z'){ if(MO_TIME(pkt)>1326409200) return MP_SEND|MP_RECV; if(MO_TIME(pkt)<1324602000) return MP_SEND|MP_RECV; }else if(MO_TYPE(pkt)>='a' && MO_TYPE(pkt)<='z'){ if(MO_TIME(pkt)>16777216) return MP_SEND; if(MO_TIME(pkt)<0) return MP_SEND; }; if(MO_TYPE(pkt)!='A' && MO_TYPE(pkt)!='a' && MO_TYPE(pkt)!='i' && MO_TYPE(pkt)!='B' && MO_TYPE(pkt)!='E' && MO_TYPE(pkt)!='F' && MO_TYPE(pkt)!='G' && MO_TYPE(pkt)!='T' ){ return MP_IGNORE|MP_RECV; }; return MP_OK; }; MPKT * meshGetMessage(uint8_t type){ int free=-1; for(int i=0;i='a' && MO_TYPE(meshbuffer[i].pkt)<='z'){ ; }else{ if (MO_TIME(meshbuffer[i].pkt)SECS_DAY) meshbuffer[i].flags=MF_FREE; }; if((mesh_sanity(meshbuffer[i].pkt)&MP_SEND)!=0){ meshbuffer[i].flags=MF_FREE; meshPanic(meshbuffer[i].pkt,i); }; }; }; }; void mesh_sendloop(void){ int ctr=0; __attribute__ ((aligned (4))) uint8_t buf[32]; int status; uint32_t rnd=0xffffffff; if(meshnice) rnd=getRandom(); nrf_config_get(&oldconfig); nrf_set_channel(MESH_CHANNEL); nrf_set_tx_mac(strlen(MESH_MAC),(uint8_t*)MESH_MAC); // Update [T]ime packet MO_TIME_set(meshbuffer[0].pkt,getSeconds()); MO_GEN_set(meshbuffer[0].pkt,meshgen); if(GLOBAL(privacy)==0) uint32touint8p(GetUUID32(),meshbuffer[0].pkt+26); else uint32touint8p(0,meshbuffer[0].pkt+26); MO_BODY(meshbuffer[0].pkt)[4]=meshnice; for (int i=0;i> 1); m = (m & 0x33333333) + ((m & 0xcccccccc) >> 2); m = (m & 0x0f0f0f0f) + ((m & 0xf0f0f0f0) >> 4); m = (m & 0x00ff00ff) + ((m & 0xff00ff00) >> 8); m = (m & 0x0000ffff) + ((m & 0xffff0000) >> 16); cnt += m; } while(--n); return cnt; } uint8_t mesh_recvqloop_work(void){ __attribute__ ((aligned (4))) uint8_t buf[32]; signed int len; len=nrf_rcv_pkt_poll_dec(sizeof(buf),buf,NULL); // Receive if(len<=0){ return 0; }; if(mesh_sanity(buf)){ meshincctr++; if((mesh_sanity(buf)&MP_RECV)!=0){ meshPanic(buf,-len); }; return 0; }; // New mesh generation? if(MO_TYPE(buf)=='T'){ if(mesh_gt(meshgen,MO_GEN(buf))){ _timet=0; meshincctr=0; meshnice=MO_BODY(buf)[4]; meshgen=MO_GEN(buf); }; }; // Discard packets with wrong generation if(meshgen != MO_GEN(buf)){ return 0; }; // Set new time iff newer if(MO_TYPE(buf)=='T'){ time_t toff=MO_TIME(buf)-((getTimer()+(600/SYSTICKSPEED))/(1000/SYSTICKSPEED)); if (toff>_timet){ // Do not live in the past. if( (_timet!=0) && (toff-_timet)>5){ meshPanic(buf,0x20); }; _timet = toff; meshincctr++; }; return 1; }; // Safety: Truncate ascii packets by 0-ing the CRC buf[MESHPKTSIZE-2]=0; // Store packet in a same/free slot MPKT* mpkt=meshGetMessage(MO_TYPE(buf)); // Propagation test if(MO_TYPE(buf)=='B'){ if(! mesh_gt(MO_BODY(mpkt->pkt)[0],MO_BODY(buf)[0]) ) return 0; (*( (uint32_t*)(MO_BODY(buf)+6) ))++; if(GLOBAL(privacy)==0) uint32touint8p(GetUUID32(),MO_BODY(buf)+20); MO_TIME_set(mpkt->pkt,0); }; #if 0 // Schnitzel if(MO_TYPE(buf)=='Z'){ mpkt->flags=MF_USED|MF_LOCK; MO_TIME_set(mpkt->pkt,getSeconds()); MO_GEN_set(mpkt->pkt,0x70); for(int i=6;ipkt[i]|=buf[i]; int score=popcount((uint32_t*)MO_BODY(mpkt->pkt),6); MPKT* reply=meshGetMessage('z'); if(MO_TIME(reply->pkt)>=score) return 1; MO_TIME_set(reply->pkt,score); strcpy((char*)MO_BODY(reply->pkt),GLOBAL(nickname)); if(GLOBAL(privacy)==0){ uint32touint8p(GetUUID32(),meshbuffer[0].pkt+26); meshbuffer[0].pkt[25]=0; }; return 1; }; #endif // only accept newer/better packets if(mpkt->flags==MF_USED) if(MO_TIME(buf)<=MO_TIME(mpkt->pkt)) return 2; #if 0 if((MO_TYPE(buf)>='A' && MO_TYPE(buf)<='C') || (MO_TYPE(buf)>='a' && MO_TYPE(buf)<='c')) meshmsg=1; #endif memcpy(mpkt->pkt,buf,MESHPKTSIZE); mpkt->flags=MF_USED; return 1; }; void mesh_recvqloop_end(void){ nrf_rcv_pkt_end(); nrf_config_set(&oldconfig); } void mesh_recvloop(void){ int recvend=M_RECVTIM/SYSTICKSPEED+getTimer(); int pktctr=0; mesh_recvqloop_setup(); do{ if( mesh_recvqloop_work() ){ pktctr++; }else{ delayms_power(10); }; }while(getTimer()MESHBUFSIZE); mesh_recvqloop_end(); }; uint8_t mesh_recvloop_plus(uint8_t state){ static int recvend=0; static int pktctr=0; if (state==0){ recvend=M_RECVTIM/SYSTICKSPEED+getTimer(); pktctr=0; mesh_recvqloop_setup(); state=1; }; if(state==1){ if( mesh_recvqloop_work() ){ pktctr++; }else{ delayms_power(10); }; if(getTimer()>recvend || pktctr>MESHBUFSIZE) state=QS_END; }; return state; }; void mesh_systick(void){ static int rcvctr=0; static int sendctr=0; if(rcvctr--<0){ push_queue_plus(&mesh_recvloop_plus); rcvctr=M_RECVINT/SYSTICKSPEED/2; rcvctr+=getRandom()%(rcvctr*2); }; if(sendctr--<0){ push_queue(&mesh_sendloop); sendctr=M_SENDINT/SYSTICKSPEED/2; sendctr+=getRandom()%(sendctr*2); }; };