Files
sics/rs232controller.c
koennecke 9737722b8c - Added hmslave.c
- Tasub only drives required motors
- Tasub handles fixed motors more gracefully
2007-05-30 11:57:50 +00:00

959 lines
20 KiB
C

/*---------------------------------------------------------------------
R S 2 3 2 C o n t r o l l e r
A general object which represents a controller connected to the network
via a terminal server. This bypasses David Maden's SerPortServer software.
Basic facilities are provided for writinga nd reading to and from the
device. For more information see the rs232controller.tex file.
copyright: see copyright.h
Mark Koennecke, October 2001
-----------------------------------------------------------------------*/
#include <stdlib.h>
#include <assert.h>
#include <errno.h>
#include <time.h>
#include "fortify.h"
#include "sics.h"
#include "splitter.h"
#include "nread.h"
#include "rs232controller.h"
/*--------------------------------------------------------------------*/
void setRS232SendTerminator(prs232 self, char *term)
{
assert(self);
if(self->sendTerminator != NULL)
{
free(self->sendTerminator);
}
if(term != NULL)
{
self->sendTerminator = strdup(term);
}
else
{
self->sendTerminator = NULL;
}
}
/*--------------------------------------------------------------------*/
void setRS232ReplyTerminator(prs232 self, char *term)
{
assert(self);
if(self->replyTerminator != NULL)
{
free(self->replyTerminator);
}
if(term != NULL)
{
self->replyTerminator = strdup(term);
}
else
{
self->replyTerminator = NULL;
}
}
/*---------------------------------------------------------------------*/
void setRS232Timeout(prs232 self, int timeout)
{
assert(self);
self->timeout = timeout;
}
/*---------------------------------------------------------------------*/
void setRS232Debug(prs232 self, int deb)
{
assert(self);
self->debug = deb;
}
/*--------------------------------------------------------------------*/
int writeRS232(prs232 self, void *data, int dataLen)
{
char *pPtr = NULL;
int iRet;
assert(self);
/*
catch an unconnected socket
*/
if(!self->pSock)
{
return NOTCONNECTED;
}
/*
do the terminator processing, if required.
If new data had to be allocated in order to add the terminator,
pPtr is not NULL. I any other case it is.
*/
if(self->sendTerminator != NULL)
{
pPtr = (char *)data;
if(strstr(pPtr,self->sendTerminator) == NULL)
{
/*
the terminator is missing. add it.
*/
pPtr = (char *)malloc((dataLen + strlen(self->sendTerminator) +2)
*sizeof(char));
if(!pPtr)
{
return BADMEMORY;
}
strcpy(pPtr,(char *) data);
strcat(pPtr,self->sendTerminator);
dataLen += strlen(self->sendTerminator);
data = pPtr;
}
else
{
pPtr = NULL;
}
}
/*
send
*/
iRet = NETWrite(self->pSock,data,dataLen);
if(self->debug > 0)
{
printf("RS232 OUT : %s",(char *)data);
if(strchr((char *)data,'\n') == NULL)
{
puts("");
}
fflush(stdout);
}
if(pPtr != NULL)
free(pPtr);
if(iRet != 1){
return BADSEND;
}
return iRet;
}
/*----------------------------------------------------------------------*/
int readRS232(prs232 self, void *data, int *dataLen)
{
long lRead;
int iRet;
size_t rLength;
assert(self);
/*
catch an unconnected socket
*/
if(!self->pSock)
{
return NOTCONNECTED;
}
/*
clean our space in order to prevent corrupted stuff
*/
memset(data,0,*dataLen);
iRet = NETAvailable(self->pSock,self->timeout);
if(iRet < 0)
{
return iRet;
}
else if(iRet == 0)
{
*dataLen = 0;
return TIMEOUT;
}
else
{
rLength = *dataLen;
lRead = recv(self->pSock->sockid, data,rLength,0);
if(lRead >= 0)
{
if(self->debug > 0)
{
printf("RS232 IN : %s",(char *)data);
if(strchr((char *)data,'\n') == NULL)
{
puts("");
}
fflush(stdout);
}
*dataLen = lRead;
return 1;
}
else
{
return (int)lRead;
}
}
/*
not reached
*/
return 0;
}
/*--------------------------------------------------------------------*/
int readRS232TillTerm(prs232 self, void *data, int *datalen){
int iRet, replylen;
assert(self);
/*
catch an unconnected socket
*/
if(!self->pSock)
{
return NOTCONNECTED;
}
memset(data,0,*datalen);
replylen = *datalen;
iRet = NETReadTillTerm(self->pSock,self->timeout,self->replyTerminator,
(char *)data, replylen);
if(self->debug > 0 && iRet > 0)
{
printf("RS232 IN/TERM : %s",(char *)data);
if(strchr((char *)data,'\n') == NULL)
{
puts("");
}
fflush(stdout);
}
if(iRet == 0)
{
if(self->debug > 0)
{
printf("RS232 IN/TERM : TIMEOUT:%s\n",(char *)data);
}
return TIMEOUT;
}
else if(iRet == -1)
{
printf("Incomplete read: %s\n", (char *)data);
return INCOMPLETE;
}
else if (iRet < 0)
{
return iRet;
}
if (*self->replyTerminator != 0) {
*datalen = strlen((char *)data);
} else {
*datalen = iRet;
}
return 1;
}
/*----------------------------------------------------------------------*/
int readRS232UntilWord(prs232 self,
char *buffer, int buflen, char *word){
time_t endTime;
int status;
int bytesRead = 0;
endTime = time(NULL) + self->timeout;
memset(buffer,0,buflen);
while(time(NULL) < endTime){
if(availableRS232(self)){
bytesRead = recv(self->pSock->sockid,buffer + bytesRead,
buflen - bytesRead - 1,0);
if(bytesRead < 0){
return BADREAD;
}
if(strstr(buffer,word) != NULL) {
return 1;
}
} else {
SicsWait(1);
}
}
return 0;
}
/*-----------------------------------------------------------------------*/
int availableRS232(prs232 self)
{
assert(self);
/*
catch an unconnected socket
*/
if(!self->pSock)
{
return NOTCONNECTED;
}
return NETAvailable(self->pSock,0);
}
/*-----------------------------------------------------------------------*/
int availableNetRS232(prs232 self)
{
int status;
assert(self);
/*
catch an unconnected socket
*/
if(!self->pSock)
{
return NOTCONNECTED;
}
if(!self->registered){
if(pServ->pReader != NULL){
NetReadRegisterUserSocket(pServ->pReader,self->pSock->sockid);
self->registered = 1;
return 0;
}
}
status = NetReadReadable(pServ->pReader,self->pSock->sockid);
NetReadResetUser(pServ->pReader, self->pSock->sockid);
return status;
}
/*------------------------------------------------------------------------*/
int transactRS232(prs232 self, void *send, int sendLen,
void *reply, int replyLen)
{
int iRet, len;
assert(self);
/*
catch an unconnected socket
*/
if(!self->pSock)
{
return NOTCONNECTED;
}
/*
if there is still data on the socket: clear it
while(availableRS232(self)){
len = replyLen;
readRS232(self,reply,&len);
}
*/
/*
write data
*/
iRet = writeRS232(self,send,sendLen);
if(iRet <= 0)
{
return BADSEND;
}
/*
read
*/
memset(reply,0,replyLen);
iRet = NETReadTillTerm(self->pSock,self->timeout,self->replyTerminator,
reply, replyLen);
if(self->debug > 0)
{
if(iRet > 0)
{
printf("RS232 IN/TRANS: %s",(char *)reply);
if(strchr((char *)reply,'\n') == NULL)
{
puts("");
}
fflush(stdout);
}
else if(iRet == 0)
{
printf("RS232 IN/TRANS: TIMEOUT\n");
}
else
{
printf("RS232 IN/TRANS/INCOMPLETE: %s",(char *)reply);
}
}
if(iRet == 0)
{
return TIMEOUT;
}
else if(iRet == -1)
{
return INCOMPLETE;
}
else
{
return iRet;
}
}
/*------------------------------------------------------------------------*/
void getRS232Error(int iCode, char *errorBuffer,
int errorBufferLen)
{
/*
the error code is either one of our errors, or an errno code from
the system
*/
switch(iCode)
{
case BADMEMORY:
strncpy(errorBuffer,
"Out of memory for appending terminators",
errorBufferLen);
break;
case NOTCONNECTED:
strncpy(errorBuffer,
"Not connected!",
errorBufferLen);
break;
case TIMEOUT:
strncpy(errorBuffer,
"Timeout or network error when reading data",
errorBufferLen);
break;
case FAILEDCONNECT:
strncpy(errorBuffer,
"Failed to connect to terminal server",
errorBufferLen);
break;
case INCOMPLETE:
strncpy(errorBuffer,"Did not find terminator in read buffer",
errorBufferLen);
break;
case BADSEND:
strncpy(errorBuffer,"Network problem: failed to send",errorBufferLen);
break;
default:
strncpy(errorBuffer,strerror(errno),
errorBufferLen);
break;
}
}
/*--------------------------------------------------------------------*/
int fixRS232Error(prs232 self, int iCode){
int i, status, read;
char buffer[8192];
switch(iCode){
case BADMEMORY:
case FAILEDCONNECT:
return 0;
break;
case INCOMPLETE:
case TIMEOUT:
/*
* try to clear possibly pending stuff
*/
for(i = 0; i < 3; i++){
if(availableRS232(self)){
read = 8192;
readRS232(self,buffer,&read);
}
}
return 1;
break;
case NOTCONNECTED:
case BADSEND:
closeRS232(self);
status = initRS232(self);
if(status > 0){
return 1;
} else {
return 0;
}
break;
}
return 0;
}
/*--------------------------------------------------------------------*/
int getRS232Timeout(prs232 self){
return self->timeout;
}
/*--------------------------------------------------------------------*/
int initRS232WithFlags(prs232 self, int flags)
{
int iRet;
assert(self);
if(self->pSock != NULL)
{
if(pServ->pReader != NULL){
NetReadRemoveUserSocket(pServ->pReader,self->pSock->sockid);
}
NETClosePort(self->pSock);
free(self->pSock);
self->pSock = NULL;
}
self->pSock = NETConnectWithFlags(self->pHost, self->iPort, flags);
if(!self->pSock){
return FAILEDCONNECT;
} else{
if(pServ->pReader != NULL){
NetReadRegisterUserSocket(pServ->pReader,self->pSock->sockid);
self->registered = 1;
}
return 1;
}
}
/*--------------------------------------------------------------------*/
int initRS232(prs232 self)
{
return initRS232WithFlags(self, 0);
}
/*--------------------------------------------------------------------*/
int initRS232Finished(prs232 self)
{
int iret;
iret = NETConnectFinished(self->pSock);
if (iret < 0) {
return FAILEDCONNECT;
} else {
return iret;
}
}
/*--------------------------------------------------------------------*/
void closeRS232(prs232 self)
{
assert(self);
if(self->pSock != NULL)
{
if(pServ->pReader != NULL){
NetReadRemoveUserSocket(pServ->pReader,self->pSock->sockid);
}
NETClosePort(self->pSock);
free(self->pSock);
self->pSock = NULL;
}
}
/*------------------------------------------------------------------*/
prs232 createRS232(char *host, int iPort)
{
prs232 pNew = NULL;
assert(iPort > 0);
/*
create data structure
*/
pNew = (prs232)malloc(sizeof(rs232));
if(!pNew)
{
return NULL;
}
memset(pNew, 0, sizeof(rs232));
pNew->pHost = strdup(host);
pNew->iPort = iPort;
pNew->sendTerminator = strdup("\r");
pNew->replyTerminator = strdup("\n");
pNew->timeout = 1000;
pNew->debug = 0;
pNew->pDes = CreateDescriptor("RS232 Controller");
if(!pNew->pDes || !pNew->pHost ||
!pNew->replyTerminator || !pNew->sendTerminator)
{
return NULL;
}
return pNew;
}
/*-------------------------------------------------------------------*/
void KillRS232(void *pData)
{
prs232 self = (prs232)pData;
if(!self)
{
return;
}
if(self->pDes)
{
DeleteDescriptor(self->pDes);
}
if(self->sendTerminator != NULL)
{
free(self->sendTerminator);
}
if(self->replyTerminator != NULL)
{
free(self->replyTerminator);
}
if(self->pSock)
{
if(pServ->pReader != NULL){
NetReadRemoveUserSocket(pServ->pReader,self->pSock->sockid);
}
NETClosePort(self->pSock);
free(self->pSock);
}
if(self->pHost)
{
free(self->pHost);
}
}
/*-------------------------------------------------------------------*/
static void KillAndFreeRS232(void *pData) {
KillRS232(pData);
free(pData);
}
/*-------------------------------------------------------------------*/
static int checkSet(SConnection *pCon, int argc, int rights)
{
if(argc < 3)
{
return 0;
}
else
{
if(SCMatchRights(pCon,rights))
{
return 1;
}
else
{
return 0;
}
}
/*
not reached
*/
return 0;
}
/*--------------------------------------------------------------------*/
static void encodeTerminator(char *result, char *terminator)
{
int i, len;
char pBuffer[10];
if(terminator == NULL)
{
result[0] = '\0';
}
len = strlen(terminator);
sprintf(pBuffer,"0x%x",(int)terminator[0]);
strcpy(result,pBuffer);
for(i = 1; i < len; i++)
{
sprintf(pBuffer,"0x%x",(int)terminator[i]);
strcat(result,pBuffer);
}
}
extern char *stptok(char *pPtr, char *pToken, int tokenLen, char *term);
/*--------------------------------------------------------------------*/
char *decodeTerminator(char *code)
{
int count = 0, icode;
char *pResult;
char *pPtr, pToken[10];
/*
max 10 terminators!
*/
pResult = (char *)malloc(10*sizeof(char));
if(!pResult)
{
return NULL;
}
memset(pResult,0,10);
pToken[0] = '0';
/*
I seem to get an empty token on the first call to stptok, this is why
I do 2 stptoks. Strange and wonderful.
*/
pPtr = stptok(code,pToken+1,9,"0");
pPtr = stptok(pPtr,pToken+1,9,"0");
while(pPtr != NULL)
{
sscanf(pToken,"%x",&icode);
pResult[count] = (char)icode;
count++;
pPtr = stptok(pPtr,pToken+1,9,"0");
}
return pResult;
}
/*--------------------------------------------------------------------*/
int RS232Action(SConnection *pCon, SicsInterp *pSics,
void *pData, int argc, char *argv[])
{
prs232 self = NULL;
char pError[256];
char pBuffer[8192], pReply[8192];
char pTerm[10];
char *pPtr = NULL;
int iRet, iRead = 8191, count, i;
self = (prs232)pData;
assert(self);
assert(pCon);
/*
check for arguments
*/
if(argc < 2)
{
sprintf(pError,"ERROR: insufficient no of arguments to %s",argv[0]);
SCWrite(pCon,pError,eError);
return 0;
}
strtolower(argv[1]);
if(strcmp(argv[1],"sendterminator") == 0)
{
if(checkSet(pCon,argc,usMugger))
{
pPtr = decodeTerminator(argv[2]);
setRS232SendTerminator(self,pPtr);
if(pPtr)
free(pPtr);
SCSendOK(pCon);
return 1;
}
else
{
encodeTerminator(pBuffer,self->sendTerminator);
sprintf(pError,"%s.sendTerminator = \"%s\"",argv[0],
pBuffer);
SCWrite(pCon,pError,eValue);
return 1;
}
}
else if(strcmp(argv[1],"debug") == 0)
{
self->debug = atoi(argv[2]);
SCSendOK(pCon);
return 1;
}
else if(strcmp(argv[1],"timeout") == 0)
{
if(checkSet(pCon,argc,usUser))
{
setRS232Timeout(self,atoi(argv[2]));
SCSendOK(pCon);
return 1;
}
else
{
sprintf(pError,"%s.Timeout = %d",argv[0],self->timeout);
SCWrite(pCon,pError,eValue);
return 1;
}
}
else if(strcmp(argv[1],"replyterminator") == 0)
{
if(checkSet(pCon,argc,usMugger))
{
memset(pTerm,0,10);
for(i = 2; i < argc;i++){
pPtr = decodeTerminator(argv[i]);
pTerm[i-2] = pPtr[0];
if(pPtr)
free(pPtr);
}
setRS232ReplyTerminator(self,pTerm);
SCSendOK(pCon);
return 1;
}
else
{
encodeTerminator(pBuffer,self->replyTerminator);
sprintf(pError,"%s.replyTerminator = \"%s\"",argv[0],
pBuffer);
SCWrite(pCon,pError,eValue);
return 1;
}
}
else if(strcmp(argv[1],"write") == 0)
{
Arg2Text(argc-2,argv+2,pBuffer,8191);
iRet = writeRS232(self,pBuffer,strlen(pBuffer));
if(iRet < 0)
{
getRS232Error(iRet,pError,255);
SCWrite(pCon,pError,eError);
return 0;
}
SCSendOK(pCon);
return 1;
}
else if(strcmp(argv[1],"read") == 0)
{
if(!availableRS232(self))
{
SCWrite(pCon,"Nothing to read!",eError);
return 1;
}
iRet = readRS232(self,pBuffer,&iRead);
if(iRet < 0)
{
getRS232Error(iRet,pError,255);
SCWrite(pCon,pError,eError);
return 0;
}
SCWrite(pCon,pBuffer,eValue);
return 1;
}
else if(strcmp(argv[1],"readchar") == 0){
if(argc < 3){
SCWrite(pCon,"ERROR: need number of chars to read",eError);
return 0;
}
iRet = Tcl_GetInt(pSics->pTcl,argv[2],&iRead);
if(iRet != TCL_OK){
SCWrite(pCon,"ERROR: failed to convert argument to number",eError);
return 0;
}
if(!availableRS232(self))
{
SCWrite(pCon,"Nothing to read!",eError);
return 1;
}
iRet = readRS232(self,pBuffer,&iRead);
if(iRet < 0)
{
getRS232Error(iRet,pError,255);
SCWrite(pCon,pError,eError);
return 0;
}
SCWrite(pCon,pBuffer,eValue);
return 1;
}
else if(strcmp(argv[1],"available") == 0)
{
iRet = availableRS232(self);
if(iRet < 0)
{
getRS232Error(iRet,pError,255);
SCWrite(pCon,pError,eError);
return 0;
}
else if(iRet == 0)
{
SCWrite(pCon,"No data pending",eValue);
return 1;
}
else
{
SCWrite(pCon,"Data available",eValue);
return 1;
}
}
else if(strcmp(argv[1],"send") == 0)
{
Arg2Text(argc-2,argv+2,pBuffer,8191);
iRet = transactRS232(self,pBuffer,strlen(pBuffer),
pReply,iRead);
if(iRet < 0)
{
getRS232Error(iRet,pError,255);
SCWrite(pCon,pError,eError);
return 0;
}
SCWrite(pCon,pReply,eValue);
return 1;
}
else if(strcmp(argv[1],"binwrite") == 0)
{
count = argc - 2;
for(i = 0; i < count; i++)
{
pBuffer[i] = (char)atoi(argv[i+2]);
}
if(self->pSock)
{
iRet = NETWrite(self->pSock,pBuffer,count);
}
else
{
iRet = NOTCONNECTED;
}
if(iRet < 0)
{
getRS232Error(iRet,pError,255);
SCWrite(pCon,pError,eError);
return 0;
}
SCSendOK(pCon);
return 1;
}
else if(strcmp(argv[1],"init") == 0)
{
iRet = initRS232(self);
if(iRet != 1)
{
sprintf(pError,"ERROR: reinitializing connection to %s at %d failed",
self->pHost, self->iPort);
SCWrite(pCon,pError,eError);
return 0;
}
else
{
SCSendOK(pCon);
return 1;
}
}
else
{
sprintf(pError,"ERROR: %s does not understand %s",argv[0], argv[1]);
SCWrite(pCon,pError,eError);
return 0;
}
return 1;
}
/*-------------------------------------------------------------------*/
int RS232Factory(SConnection *pCon, SicsInterp *pSics,
void *pData, int argc, char *argv[])
{
prs232 pNew = NULL;
int iRet, status;
char pError[256];
if(argc < 4)
{
SCWrite(pCon,"ERROR: insufficient no of arguments to RS232Factory",
eError);
return 0;
}
/*
create data structure and open port
*/
pNew = createRS232(argv[2], atoi(argv[3]));
if(!pNew)
{
SCWrite(pCon,"ERROR: out of memory in RS232Factory",eError);
return 0;
}
status = initRS232(pNew);
if(status != 1)
{
sprintf(pError,"ERROR: failed to connect to %s at port %d",
pNew->pHost, pNew->iPort);
SCWrite(pCon,pError,eError);
}
/*
create the command
*/
iRet = AddCommand(pSics,argv[1],RS232Action, KillAndFreeRS232, pNew);
if(!iRet)
{
sprintf(pError,"ERROR: duplicate command %s not created", argv[1]);
SCWrite(pCon,pError,eError);
KillAndFreeRS232(pNew);
return 0;
}
return 1;
}