QString MainWindow::readIni() { QSettings settings("settings.ini",QSettings::IniFormat); QString path=settings.value("DataStorage/Path","empty").toString(); if(path=="empty") { path=createIni(); } path+="/"; return path; }
int main(int argc, char *argv[]) { modbus_t *ctx_tcp; modbus_t *ctx_serial; uint8_t *query; uint8_t *raw_query; uint8_t *raw_reply; int header_length; int socket; int rc; modbus_mapping_t *mb_mapping; modbus_mapping_t *local_mb_mapping; const int RAW_REQ_LENGTH = 10; int RTU; int exitFlag = 0; // Set to non-zero to exit. // int opt=1; int cnt=1; // int tty_len=0; int baud_rate=9600; char parity='N'; int length=8; int stop_bits=2; char *tty=(char *)NULL; char *ip=(char *)NULL; int port=1502; int verbose=1; int io_address=0; int tmp_address=0; uint16_t *localInputRegisters; int localInputRegistersSize; uint16_t *localHoldingRegisters; int localHoldingRegistersSize; int holdingShmKey=0; int holdingShmId=0; int holdingSemKey=0; int holdingSemId=0; int inputShmKey=0; int inputShmId=0; int inputSemKey=0; int inputSemId=0; int i; unsigned char *tmp; int ch; // for getopt char *iniFile=(char *)NULL; char *ptr; int useSharedInput=0; int useSharedHolding=0; int modbusDebug = 0; int localRTU = 0; int serialRTUenabled = 0; char *iniFileName[] = { "/etc/newServer.ini", "/usr/local/etc/newServer.ini", "./newServer.ini" }; while ((ch = getopt(argc,argv,"b:c:h?i:p:P:t:v")) != -1) { switch (ch) { case 'c': if ( iniFile != (char *)NULL) { free(iniFile); } iniFile = strsave( optarg ); if(verbose) printf("Set ini file to %s\n",iniFile); break; case 'n': // baud rate baud_rate=atoi(optarg); break; case 'h': case '?': usage(); exit(0); break; case 'i': if ( ip != (char *)NULL) { free(ip); } ip=strsave( optarg ); break; case 'p': port=atoi( optarg ); break; case 'P': parity=optarg[0]; if ('E' != parity && 'O' != parity && 'N' != parity) { fprintf(stderr,"Invalid parity %c, exiting.\n",parity); exit(-1); } break; case 't': // serial port if ( tty != (char *)NULL) { free( tty ); } tty=strsave( optarg ); break; case 'v': verbose=1; break; default: break; } } if (iniFile == (char *)NULL) { for(i=0;(i<3 && (iniFile == (char *)NULL)) ;i++) { if ( verbose) { printf("%d:%s\n",i, iniFileName[i]); } if( 0 == access( iniFileName[i],R_OK)) { if(verbose ) { printf("\tFile exists\n"); } iniFile = strsave( iniFileName[i]); } } } if (iniFile == (char *)NULL ) { printf("No existing ini file.\n"); // Create an default file iniFile=strsave("./newServer.ini"); printf("\tCreating new ini file %s\n", iniFile); createIni( iniFile ); // exit(0); } ini = iniparser_load( iniFile ); ptr= iniparser_getstring(ini,"network:IP","127.0.0.1"); if((char *)NULL == ip) { ip=strsave(ptr); } if (0 == port) { port = iniparser_getint(ini,"network:port",1502); } if (-1 == verbose) { verbose = iniparser_getboolean(ini,"system:verbose",0); } localHoldingRegistersSize = iniparser_getint(ini,"system:holding_reg_size", LOCAL_HOLDING_REGISTERS); localInputRegistersSize = iniparser_getint(ini,"system:input_reg_size", LOCAL_INPUT_REGISTERS); useSharedInput = iniparser_getboolean(ini,"system:share_input",0); useSharedHolding = iniparser_getboolean(ini,"system:share_holding",0); modbusDebug = iniparser_getboolean(ini,"modbus:debug",0); localRTU = iniparser_getint(ini,"modbus:LOCAL_RTU",255); serialRTUenabled = iniparser_getboolean(ini,"modbus:RTU",0); if( serialRTUenabled ) { if( (char *) NULL == tty) { tty = iniparser_getstring(ini,"modbus:tty","/dev/ttyUSB0"); } } else { tty = (char *)NULL; } if(verbose) { printf("\n\t\tSettings\n"); printf( "\t\t========\n\n"); printf("\tBuild Date\t:%s\n",__DATE__); printf("\tAddress\t\t:%s\n",ip); printf("\tNetwork port\t:%d\n",port); printf("\tSerial port\t:%s\n",tty); printf("\tBaud rate\t:%d\n",baud_rate); printf("\tParity\t\t:%c\n",parity); } else { printf("\nStarted.\n"); } if (serialRTUenabled ) { if(access(tty,F_OK) == 0) { printf("Serial port %s exists.....\n",tty); // // Now test that it is readable, writable. // } else { printf("Serial port %s does NOT exist\n",tty); printf("Disabling serial RTU.\n"); free( tty ); tty = (char *)NULL; serialRTUenabled=0; } } query = malloc(MODBUS_TCP_MAX_ADU_LENGTH); raw_query = malloc(RAW_REQ_LENGTH); raw_reply = malloc(MODBUS_TCP_MAX_ADU_LENGTH); mb_mapping = modbus_mapping_new( MODBUS_MAX_BITS, // Coils MODBUS_MAX_BITS, // Discrete inputs MODBUS_MAX_REGISTERS, // Holding registers MODBUS_MAX_REGISTERS); // Input registers local_mb_mapping = modbus_mapping_new( NULL, // Coils NULL, // Discrete inputs LOCAL_HOLDING_REGISTERS, // Holding registers LOCAL_INPUT_REGISTERS); // Input registers // // Setup the memory for the holding registers. // if ( useSharedHolding ) { if (verbose) printf("Share the Holding regsisters R/W\n"); localHoldingRegisters = local_mb_mapping->tab_registers; holdingShmKey = iniparser_getint(ini, "system:holding_reg_key",110); holdingSemKey = iniparser_getint(ini, "system:holding_reg_sema",111); holdingShmId = shmget(holdingShmKey, localHoldingRegistersSize, IPC_CREAT | 0600 ); // change to 0400 if (holdingShmId < 0) { perror("Holding Reg shmget"); exit(3); } local_mb_mapping->tab_registers = (uint16_t *)shmat(holdingShmId,NULL,0); if (local_mb_mapping->tab_registers < 0) { perror("Holding Reg shmat"); exit(4); } /* * Create the seamphore */ holdingSemId = semget( holdingSemKey, 1, IPC_CREAT | 0600 ); if (-1 == holdingSemId) { perror("Input reg semaphore"); exit(2); } /* * Shared memory created, semaphore created (default is locked) * release the seamphore ! */ rc = semctl(holdingSemId, 0, SETVAL, 1); } // // Setup the memory for the shared input registers. // if ( useSharedInput ) { if (verbose) printf("Share the Input regsisters R/W\n"); localInputRegisters = local_mb_mapping->tab_input_registers; // Save the current pointer inputShmKey = iniparser_getint(ini, "system:input_reg_key",110); inputSemKey = iniparser_getint(ini, "system:input_reg_sema",111); inputShmId = shmget(inputShmKey, localInputRegistersSize, IPC_CREAT | 0600 ); // change to 0400 if (inputShmId < 0) { perror("Input Reg shmget"); exit(3); } local_mb_mapping->tab_input_registers = (uint16_t *)shmat(inputShmId,NULL,0); if ( local_mb_mapping->tab_input_registers < 0) { perror("Input Reg shmat"); exit(1); } /* * create the seamphore */ inputSemId = semget( inputSemKey, 1, IPC_CREAT | 0600 ); if (-1 == inputSemId) { perror("Input reg semaphore"); exit(2); } /* * Shared memory created, semaphore created (default is locked) * release the seamphore ! */ rc = semctl(inputSemId, 0, SETVAL, 1); } memset( localInputRegisters,0x00, LOCAL_INPUT_REGISTERS); memset( localHoldingRegisters,0x00, LOCAL_HOLDING_REGISTERS); local_mb_mapping->tab_input_registers[0]=0xaa55; ctx_tcp = modbus_new_tcp(ip, port); if ( tty != (char *)NULL ) { ctx_serial = modbus_new_rtu(tty, baud_rate, parity, length, stop_bits); } while ( !exitFlag ) { header_length = modbus_get_header_length(ctx_tcp); if(verbose) { modbus_set_debug(ctx_tcp, TRUE); // modbus_set_debug(ctx_serial, TRUE); } do { socket = modbus_tcp_listen(ctx_tcp, 5); if (socket < 0) { printf("%04d: Socket in use.\n",cnt++); sleep(1); } } while(socket < 0); // printf("modbus_tcp_listen =%d\n",socket); if (socket == -1) { printf("\t: %s\n", modbus_strerror(errno)); } rc=modbus_tcp_accept(ctx_tcp, &socket); // printf("modbus_tcp_accept rc=%d\n",rc); if (rc == -1) { fprintf(stderr,"FATAL ERROR: %s\n", modbus_strerror(errno)); exit(-1); } while ( 1 ) { do { if(verbose) { printf("Recieve.\n"); } rc = modbus_receive(ctx_tcp, query); // printf("rc=%d\n",rc); /* Filtered queries return 0 */ } while (rc == 0); if (-1 == rc) { if( verbose ) { printf("Client disconnected.\n"); } break; } RTU=query[header_length - 1]; io_address= ( query[header_length+1] << 8) + (query[header_length+2] & 0xff); printf("IO Address=%04x\n",io_address); if (verbose) { if( localRTU == RTU ) { printf("It's for me !\n"); } else { printf("RTU Id: %02d\n",RTU); } printf("IO Address=0x%04x\n",io_address); printf("Seq Number=0x%04x\n", query[1]); } if( localRTU == RTU ) { // // Local mapping // int len; uint16_t data; int cnt=0; // Need to add address range checks, and return an error // for an invalid function. switch (query[header_length]) { case 0x03: // read holding registers case 0x04: // read input registers rc = modbus_reply(ctx_tcp, query, rc, local_mb_mapping); break; case 0x06: //write single register swab(&query[8],&len,2); swab(&query[10],&data,2); printf("Data=%02d\n",data); local_mb_mapping->tab_registers[io_address]=data; rc = modbus_reply(ctx_tcp, query, rc, local_mb_mapping); break; case 0x10: // Write multiple registers // len=query[7]; swab(&query[10],&len,2); printf("Length=%02d\n",len); for (i=13; i< (13 + len ); i +=2) { data= (query[i] << 8) + (query[i+1] & 0xff); printf("%04x\n",data); local_mb_mapping->tab_registers[ io_address + cnt++ ] = data; } rc = modbus_reply(ctx_tcp, query, rc, local_mb_mapping); break; default: break; } } else { if(serialRTUenabled) { modbus_set_slave( ctx_serial,RTU); if( -1 == modbus_connect( ctx_serial)) { fprintf(stderr, "Connection failed: %s\n",modbus_strerror(errno)); } switch( query[header_length] ) { case 0x06: printf("Write Single Register %d\n",RTU); memcpy(raw_query,&query[header_length-1], 6); rc=modbus_send_raw_request( ctx_serial,raw_query,6); modbus_receive_confirmation(ctx_serial, raw_reply); rc = modbus_reply(ctx_tcp, query, rc, mb_mapping); break; case 0x03: case 0x04: printf("Read Registers %d\n",RTU); break; case 0x10: { int len=0; // modbus_set_debug(ctx_serial, TRUE); printf("Write multiple registers %d\n",RTU); len=query[header_length - 2]; printf("Length %d\n",len); memcpy(raw_query,&query[header_length-1], len); rc=modbus_send_raw_request( ctx_serial,raw_query,len); if( -1 == rc) { printf("modbus_send_raw_request: %s\n",modbus_strerror(errno)); } else { modbus_receive_confirmation(ctx_serial, raw_reply); rc = modbus_reply(ctx_tcp, query, rc, mb_mapping); printf("rc = %d\n",rc); // modbus_set_debug(ctx_serial, FALSE); } } break; } if( (0x03 == query[header_length]) || (0x04 == query[header_length])) { int len; if(verbose) { switch (query[header_length]) { case 0x03: printf("Read Holding Registers.\n"); break; case 0x04: printf("Read Input Registers.\n"); break; default: break; } } memcpy(raw_query,&query[header_length-1], 6); modbus_set_slave( ctx_serial, query[header_length-1]); if( -1 == modbus_connect( ctx_serial)) { fprintf(stderr, "Connection failed: %s\n",modbus_strerror(errno)); } rc=modbus_send_raw_request( ctx_serial,raw_query,6); if( -1 == rc ) { printf("modbus_send_raw_request: %s\n",modbus_strerror(errno)); } modbus_receive_confirmation(ctx_serial, raw_reply); /* * This next loop swaps bytes in words. * If this is built and running on a little endian * machine (most machines are these days) * Then this needs to be done. * If on a big endian machine (M68000 family) just * comment this out. * * It might be worth having a command line switch, * detect endianness. * */ switch (query[header_length]) { case 0x03: printf("Read Holding Registers.\n"); tmp=(unsigned char *)mb_mapping->tab_registers; break; case 0x04: printf("Read Input Registers.\n"); tmp=(unsigned char *)mb_mapping->tab_input_registers; break; default: break; } tmp_address = (io_address*2); for(i=0;i<raw_reply[2];i=i+2 ) { tmp[tmp_address+i]=raw_reply[i+4]; tmp[tmp_address+i+1]=raw_reply[i+3]; } for(i=0;i<10;i++) { if (verbose) { printf("!%02x!",raw_reply[i]); } } printf("\n"); for(i=0;i<10;i++) { if (verbose) { printf("+%02x+",tmp[i]); } } printf("\n"); // // compute packet length by getting the data length // and then add 2. // Then overwrite the outbound data. // len=raw_reply[2] + 3; rc = modbus_reply(ctx_tcp, query, rc, mb_mapping); // memcpy( mb_mapping->tab_input_registers, &raw_reply[3], raw_reply[2]); // MODBUS_SET_INT16_TO_INT8(query, header_length + 3, UT_REGISTERS_NB_SPECIAL - 1); } } } // printf("Reply with an invalid TID or slave\n"); // rc=modbus_reply_exception(ctx_tcp, query, 1); // rc=modbus_send_raw_request(ctx_tcp, raw_reply, 10 * sizeof(uint8_t)); // rc = modbus_reply(ctx_tcp, query, rc, mb_mapping); } if( verbose ) { printf("Tidying up\n"); } close(socket); usleep( 1000 ); // wait 1 ms } printf("simpleServer exiting.\n"); modbus_close(ctx_tcp); modbus_free(ctx_tcp); free(query); modbus_mapping_free(mb_mapping); return(0); }