OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [tags/] [start/] [gdb-5.0/] [utils/] [amd-udi/] [montip/] [checksum.c] - Rev 1765

Compare with Previous | Blame | View Log

static char _[] = "@(#)checksum.c	5.25 93/10/27 15:11:54, Srini, AMD.";
/******************************************************************************
 * Copyright 1991 Advanced Micro Devices, Inc.
 *
 * This software is the property of Advanced Micro Devices, Inc  (AMD)  which
 * specifically  grants the user the right to modify, use and distribute this
 * software provided this notice is not removed or altered.  All other rights
 * are reserved by AMD.
 *
 * AMD MAKES NO WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, WITH REGARD TO THIS
 * SOFTWARE.  IN NO EVENT SHALL AMD BE LIABLE FOR INCIDENTAL OR CONSEQUENTIAL
 * DAMAGES IN CONNECTION WITH OR ARISING FROM THE FURNISHING, PERFORMANCE, OR
 * USE OF THIS SOFTWARE.
 *
 * So that all may benefit from your experience, please report  any  problems
 * or  suggestions about this software to the 29K Technical Support Center at
 * 800-29-29-AMD (800-292-9263) in the USA, or 0800-89-1131  in  the  UK,  or
 * 0031-11-1129 in Japan, toll free.  The direct dial number is 512-462-4118.
 *
 * Advanced Micro Devices, Inc.
 * 29K Support Products
 * Mail Stop 573
 * 5900 E. Ben White Blvd.
 * Austin, TX 78741
 * 800-292-9263
 *****************************************************************************
 *      Engineer: Srini Subramanian.
 *****************************************************************************
 **
 **       This file contains code for intercepting failed messages
 **	 due to serial transmission errors.  The code logically
 **	 resides between the messages system and the character
 ** 	 based serial driver.  Messages with invalid checksums, or
 ** 	 other communication errors are retried.
 *****************************************************************************
 */
 
#include <stdio.h>
#include <string.h>
#include "types.h"
#include "memspcs.h"
#include "messages.h"
#include "mtip.h"
#include "tdfunc.h"
 
#ifdef MSDOS
#include <conio.h>
#endif
 
#define ACK_HDR_CODE -1
#define CHECKSUM_PASS 0
#define CHECKSUM_FAIL -1
 
extern	int	MessageRetries;
extern	unsigned int	TimeOut;
extern	unsigned int	BlockCount;
#ifdef __hpux
static volatile int	bcount;
#else
static int	bcount;
#endif
 
extern	int	use_parport;
 
#ifdef MSDOS
INT32	par_write PARAMS ((char *, INT32));
#endif
 
void   endian_cvt PARAMS((union msg_t *, int));
void   send_nack PARAMS((INT32 port_base));
 
extern	FILE	*MsgFile;	/* for logging error retries */
 
struct	ack_msg_t {
  INT32		code;
  INT32		passfail;
};
union  ack_msg_buf_t {
   struct ack_msg_t  ack_msg;
   unsigned char     buf[(2*sizeof(INT32))];
};
 
 
INT32
msg_send_serial(msg_ptr, port_base)
   union  msg_t  *msg_ptr;
   INT32  port_base;
   {
   INT32 result, i, ack, comm_err; 
   UINT32 checksum;
   int		retries;
   unsigned int		timeout;
   INT32 	Rx_ack[2];
 
   INT32 header_size = (2 * sizeof(INT32));
 
   BYTE  *bfr_ptr = (BYTE *) msg_ptr;
 
   /* Save length before doing endian conversion */
   INT32 length = msg_ptr->generic_msg.length;
   INT32 total_length;
 
   /*
    * MiniMON29K release 2.1 has new Communications Interface module
    * which does not expect the checksum to be aligned on a word 
    * boundary. It expects the checksum to immediately follow the
    * end of the message body.
    * The old handler aligned the checksum on the next word boundar after
    * the message body, but did _not_ update the message length field.
    * That caused problems when one end of the Communications gets
    * changed.
    */
   if (((tip_target_config.version >> 24) & 0xf) > 5) { /* new comm handler */
   } else { /* old comm handler */
      /* round length up to even word */
      if ((length & 3) != 0) {	/* round up to word boundary */
   	   length = length + 3;
   	   length = length & 0xfffffffc;
      };
   }
 
   total_length = header_size + length;
 
   /* Endian conversion */
   if (tip_target_config.TipEndian != tip_target_config.P29KEndian)
      endian_cvt(msg_ptr, OUTGOING_MSG);
 
   /* calc checksum for msg */
   checksum = 0;    
   for (i=0; i < total_length; i++)
     checksum = checksum + bfr_ptr[i];
 
   /* Append checksum to the end of the message. Do not update the
    * "length" field of the message header.
    */
   bfr_ptr[total_length] = (BYTE) ((checksum >> 24) & 0xff);
   bfr_ptr[total_length+1] = (BYTE) ((checksum >> 16) & 0xff);
   bfr_ptr[total_length+2] = (BYTE) ((checksum >> 8) & 0xff);
   bfr_ptr[total_length+3] = (BYTE) ((checksum >> 0) & 0xff);
   /* send msg */
   retries = 0;
   do  {
	retries = retries + 1;
	comm_err = (INT32) 0;
 
	/* send msg */ 
        result = send_bfr_serial(bfr_ptr, total_length+4, /* 4 for checksum*/
					port_base, &comm_err);
	if (comm_err != (INT32) 0) {
	  (void) reset_comm_serial ((INT32) -1, (INT32) -1);
	  return ((INT32) MSGRETRY);
	}
	if (result != (INT32) 0)
	    return((INT32) FAILURE);
 
	/* get ack */
	timeout = 0;
	result = (INT32) -1;
	comm_err = (INT32) 0;
	while ((timeout < TimeOut) && (result == (INT32) -1) 
					  && (comm_err == (INT32) 0)) {
	/* Poll for user interrupt */
	   SIGINT_POLL
	   timeout=timeout+1;
           result = recv_bfr_serial((BYTE *) Rx_ack, (2 * sizeof(INT32)), 
					BLOCK, port_base, &comm_err);
#ifndef MSDOS
	   /* printf("ack wait timeout=0x%lx\n", timeout); */
	   if (result == (INT32) -1)
	     for (bcount = 0; bcount < BlockCount; bcount++);
#endif
	}
 
	if (comm_err != (INT32) 0) {
	     reset_comm_serial((INT32) -1, (INT32) -1);
	     return ((INT32) MSGRETRY);
	}
	/* Poll for user interrupt */
	   SIGINT_POLL
	/* check if timed out */
	if (timeout >= TimeOut) {
	 if (MsgFile) {
	   fprintf(MsgFile,"Timed out before ACK received. Reset comm. retries=%d timeout=%ld\n",retries, timeout);
     	   fflush(MsgFile);
	  }
	  (void) reset_comm_serial((INT32) 0, (INT32) 0);
	  continue;
	}
 
	ack = (INT32) Rx_ack[1];
 
	/* endian convert Ack */
    	if (tip_target_config.TipEndian != tip_target_config.P29KEndian)
       			        tip_convert32((BYTE *) &ack);
 
	if (Rx_ack[0] == (INT32) 0xFFFFFFFF && ack != CHECKSUM_FAIL) { 
		return(0);		/* successful send */
		}
	else {
	  (void) reset_comm_serial((INT32) 0, (INT32) 0);
  		if (MsgFile) {	/* log the error */
     	 		fprintf(MsgFile, 
			  "\n** Checksum: Nack Received, Resending.\n");
     			fflush(MsgFile);
  			};
		}   
 
   } while ( retries < MessageRetries);
   return ((INT32) FAILURE);
}
 
INT32
msg_recv_serial(msg_ptr, port_base, Mode)
   union  msg_t  *msg_ptr;
   INT32  port_base;
   INT32  Mode;	/* Block or NonBlock */
   {
   union  ack_msg_buf_t  AckMsg;
   UINT32 checksum_calc, checksum_recv;
   INT32 i, result;
   INT32 comm_err;
   INT32 ack_hdr;
   BYTE  *bfr_ptr;
   INT32 header_size;
   INT32 length, total_length;
 
again:
	/* Poll for user interrupt */
	   SIGINT_POLL
 
   comm_err = (INT32) 0;
   ack_hdr = (INT32) ACK_HDR_CODE;
   bfr_ptr = (BYTE *) msg_ptr;
   header_size = (2 * sizeof(INT32));
 
   /* recv header - if available */ 
   if (Mode == NONBLOCK) {
#ifndef MSDOS
	     for (bcount = 0; bcount < BlockCount; bcount++);
#endif
     result = recv_bfr_serial(bfr_ptr, header_size, Mode,
					port_base, &comm_err);
	      /* printf("nbread: result = 0x%lx Mode=0x%lx\n", result, Mode); */
   } else {
	     /* printf("bread: header_size = %d Mode=0x%lx\n", header_size, Mode); */
     result = recv_bfr_serial(bfr_ptr, header_size, Mode,
					port_base, &comm_err);
     if (result == (INT32) -1) {
#ifndef MSDOS
	     for (bcount = 0; bcount < BlockCount; bcount++);
#endif
	     goto again;
     }
   }
   if (comm_err != (INT32) 0) {
	  (void) reset_comm_serial ((INT32) -1, (INT32) -1);
	  send_nack(port_base);
	  goto again;
   }
   if (result != (INT32) 0)
	    return((INT32) FAILURE);
 
	/* Poll for user interrupt */
	   SIGINT_POLL
   /*
    * Before computing the length here, we should make sure that we have
    * received a valid (defined) MiniMON29K message by checking the
    * Message Code field. Otherwise, a lousy stream of bytes could send this
    * to a toss waiting for an unknown number of bytes.
    * But we hope none of those things would happen here!
    */
   result = msg_ptr->generic_msg.code;
   if (tip_target_config.TipEndian != tip_target_config.P29KEndian)
        	tip_convert32((BYTE *) &result);
   if ((result < (INT32) 0) || (result > 101)) {
	(void) reset_comm_serial ((INT32) -1, (INT32) -1);
	send_nack(port_base);
	goto again;	/* retry */
   }
   /* Message header received.  Save message length. */
   length = msg_ptr->generic_msg.length;
   if (tip_target_config.TipEndian != tip_target_config.P29KEndian)
        	tip_convert32((BYTE *) &length);
 
   /*
    * MiniMON29K release 2.1 has new Communications Interface module
    * which does not expect the checksum to be aligned on a word 
    * boundary. It expects the checksum to immediately follow the
    * end of the message body.
    * The old handler aligned the checksum on the next word boundar after
    * the message body, but did _not_ update the message length field.
    * That caused problems when one end of the Communications gets
    * changed.
    */
   if (((tip_target_config.version >> 24) & 0xf) > 5) { /* new comm handler */
   } else { /* old comm handler */
      /* round length up to even word */
      if ((length & 3) != 0) {	
   	   length = length + 3;
   	   length = length & 0xfffffffc;
      }
   }
 
   /* committed now - recv rest of msg and checksum */
   comm_err = (INT32) 0;
   result = (INT32) 0;
   if (length >= 0) {
   	result = recv_bfr_serial(bfr_ptr + header_size, length+4,/* +4 */
				BLOCK, port_base, &comm_err);
   }
 
   if (comm_err != (INT32) 0) {
	  (void) reset_comm_serial ((INT32) -1, (INT32) -1);
	  send_nack(port_base);
	  goto again;	/* retry */
   }
   if (result != (INT32) 0) {
	  (void) reset_comm_serial ((INT32) -1, (INT32) -1);
	    send_nack(port_base);
	    goto again;	/* retry */
    }
 
 
   /* Do endian conversion */
   if (tip_target_config.TipEndian != tip_target_config.P29KEndian)
      	endian_cvt(msg_ptr, INCOMING_MSG);
 
   /* calc checksum for msg */
   checksum_calc = 0;    
   total_length = header_size + length;
   for (i=0; i < total_length; i++)
      checksum_calc = checksum_calc + ((UINT32) bfr_ptr[i]);
 
   checksum_recv = (UINT32) 0;
   checksum_recv = (UINT32) (checksum_recv | ((UINT32) bfr_ptr[total_length] << 24));
   checksum_recv = (UINT32) (checksum_recv | ((UINT32) bfr_ptr[total_length+1] << 16));
   checksum_recv = (UINT32) (checksum_recv | ((UINT32) bfr_ptr[total_length+2] << 8));
   checksum_recv = (UINT32) (checksum_recv | ((UINT32) bfr_ptr[total_length+3] << 0));
 
	/* Poll for user interrupt */
	   SIGINT_POLL
   /* Compare Checksums */
   if (checksum_calc != checksum_recv) {
	  (void) reset_comm_serial ((INT32) -1, (INT32) -1);
	send_nack(port_base);
	goto	again;	/* retry */
	}
 
   /* send checksum hdr & ack */
   AckMsg.ack_msg.code = (INT32) ACK_HDR_CODE;
   AckMsg.ack_msg.passfail = CHECKSUM_PASS;
   result = (INT32) 0;
   comm_err = (INT32) 0;
#ifdef MSDOS
   if (use_parport) {
     result = par_write ((char *) AckMsg.buf, 8);
   } else {
     result = send_bfr_serial((BYTE *) AckMsg.buf, 2 * sizeof(INT32),
				port_base, &comm_err);
     if (comm_err != (INT32) 0) {
	  (void) reset_comm_serial ((INT32) -1, (INT32) -1);
	  send_nack(port_base);
	  goto	again;	/* retry */
     }
     if (result != (INT32) 0)  {
       if (MsgFile) {
         fprintf(MsgFile, "Couldn't send checksum to acknowledge.\n");
         fflush (MsgFile);
       }
       (void) reset_comm_serial((INT32) -1, (INT32) -1);
       send_nack(port_base);
       goto	again;	/* retry */
     }
   }
#else
     result = send_bfr_serial((BYTE *) AckMsg.buf, 2 * sizeof(INT32),
				port_base, &comm_err);
     if (comm_err != (INT32) 0) {
	  (void) reset_comm_serial ((INT32) -1, (INT32) -1);
	  send_nack(port_base);
	  goto	again;	/* retry */
     }
     if (result != (INT32) 0)  {
       if (MsgFile) {
         fprintf(MsgFile, "Couldn't send checksum to acknowledge.\n");
         fflush (MsgFile);
       }
       (void) reset_comm_serial((INT32) -1, (INT32) -1);
       send_nack(port_base);
       goto	again;	/* retry */
     }
#endif
 
   return(msg_ptr->generic_msg.code);   /* passed */
}
 
void
SendACK(port_base)
   INT32  port_base;
   {
   union ack_msg_buf_t  AckMsg;
   INT32 result, comm_err;
   INT32 ack_hdr = (INT32) ACK_HDR_CODE;
   INT32 ack = CHECKSUM_FAIL;
 
   AckMsg.ack_msg.code = (INT32) ACK_HDR_CODE;
   AckMsg.ack_msg.passfail = CHECKSUM_PASS;
   result = (INT32) 0;
   comm_err = (INT32) 0;
#ifdef MSDOS
   if (use_parport) {
     result = par_write((char *) AckMsg.buf, 8);
     return;
   } else {
      result = send_bfr_serial((BYTE *) AckMsg.buf, 2*sizeof(INT32),
				    port_base, &comm_err);
      if ((result != (INT32) 0) || (comm_err != (INT32) 0)) {
        if (MsgFile) {
          fprintf(MsgFile, "Couldn't send ACK to remote.\n");
          fflush (MsgFile);
        }
        return ;
      }
   }
#else
      result = send_bfr_serial((BYTE *) AckMsg.buf, 2*sizeof(INT32),
				    port_base, &comm_err);
      if ((result != (INT32) 0) || (comm_err != (INT32) 0)) {
        if (MsgFile) {
          fprintf(MsgFile, "Couldn't send ACK to remote.\n");
          fflush (MsgFile);
        }
        return ;
      }
#endif
} 
 
void
send_nack(port_base)
   INT32  port_base;
   {
   union ack_msg_buf_t  NAckMsg;
   INT32 result, comm_err;
   INT32 ack_hdr = (INT32) ACK_HDR_CODE;
   INT32 ack = CHECKSUM_FAIL;
 
   /* eat up any incoming characters */
   result = reset_comm_serial(port_base, port_base); 	/* reset buffer */
 
   if (MsgFile) {	/* log the error */
  	fprintf(MsgFile, 
	  "\n** Checksum: Receive failed, sending Nack.\n");
     	fflush(MsgFile);
  	};
 
	/* Poll for user interrupt */
	   SIGINT_POLL
   NAckMsg.ack_msg.code = (INT32) ACK_HDR_CODE;
   NAckMsg.ack_msg.passfail = CHECKSUM_FAIL;
   result = (INT32) 0;
   comm_err = (INT32) 0;
#ifdef MSDOS
   if (use_parport) {
     result = par_write((char *) NAckMsg.buf, 8);
     return;
   } else {
      result = send_bfr_serial((BYTE *) NAckMsg.buf, 2*sizeof(INT32),
				    port_base, &comm_err);
      if ((result != (INT32) 0) || (comm_err != (INT32) 0)) {
        if (MsgFile) {
          fprintf(MsgFile, "Couldn't send NACK to remote.\n");
          fflush (MsgFile);
        }
        return ;
      }
   }
#else
      result = send_bfr_serial((BYTE *) NAckMsg.buf, 2*sizeof(INT32),
				    port_base, &comm_err);
      if ((result != (INT32) 0) || (comm_err != (INT32) 0)) {
        if (MsgFile) {
          fprintf(MsgFile, "Couldn't send NACK to remote.\n");
          fflush (MsgFile);
        }
        return ;
      }
#endif
} 
 

Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.