*** empty log message ***
[h316.git] / pc-tools / ldc2 / src / tape_block.cpp
index 0d011ebb0392b59635a3becec22f5c71eb3776c6..b6fc25b49183b5b80ebe296ce2fabbd9e477e03c 100644 (file)
@@ -1,10 +1,21 @@
+
+
+#include <string>
+#include <vector>
+
 #include <stdlib.h>
 #include <unistd.h>
 #include <string.h>
 #include <stdio.h>
 
 #include "tape_block.hh"
+
+#include "silent_code.hh"
+#include "hw_constants.hh"
+
+#include "eot_block.hh"
 #include "data_block.hh"
+#include "broken_block.hh"
 #include "data_block_0.hh"
 #include "data_block_0_0.hh"
 #include "data_block_0_1.hh"
 #include "data_block_0_54.hh"
 #include "data_block_0_60.hh"
 #include "data_block_0_50.hh"
-
-#include "eot_block.hh"
-#include "discard_block.hh"
-#include "silent_code.hh"
-#include "hw_constants.hh"
+#include "data_block_0_64.hh"
+#include "data_block_1.hh"
+#include "data_block_2.hh"
+#include "data_block_3.hh"
+#include "data_block_4.hh"
+#include "data_block_5.hh"
+#include "data_block_6.hh"
+#include "data_block_7.hh"
 
 using namespace std;
 
@@ -50,8 +64,7 @@ tape_block::tape_block(tape_block & org){
  */
 void tape_block::operator=(tape_block &org){
   block_type=org.block_type;
-  init_state=org.init_state;
-  data_read=org.data_read; 
+  discarded_bytes=org.discarded_bytes;
   raw_data=0;  
   raw_size=org.raw_size;  
   word_data=0;
@@ -66,17 +79,6 @@ void tape_block::operator=(tape_block &org){
    }
 }
 
-/***************************************************************/
-/*!
- *\brief Query the block's state.
- *
- * If the state is not TBS_OK, the block must be discarded.
- *\return The block's initialisation state
- */
-tape_block::tb_state_t tape_block::get_state(){
-  return init_state;
-}
-
 /***************************************************************/
 /*!
  *\brief Determine the block's type
@@ -84,7 +86,6 @@ tape_block::tb_state_t tape_block::get_state(){
  * This routine returns the block's type.\\
  *\retval TBT_DATA if it's a data block
  *\retval TBT_EOT if it's an end of tape mark
- *\retval TBR_DISCARD if the block is unusable
  *
  *\note This function is virtual. That means that if the tape block in
  * question is of the subtype data_block which overwrites this method,
@@ -129,6 +130,16 @@ int tape_block::get_raw_size(){
   return raw_size;
 }
 
+/***************************************************************/
+/*!
+ *\brief Get amount of bytes discarded before beginning
+ *       of block sensed.
+ *\return Length of unusable tape before block start delimiter. 
+ */
+int tape_block::get_discarded_bytes(){
+  return discarded_bytes;
+}
+
 /***************************************************************/
 /*!
  *\brief Access internal raw data buffer
@@ -143,13 +154,82 @@ unsigned char * tape_block::get_raw_data(){
 
 /***************************************************************/
 /*!
- *\brief Static factory method with file descriptor support.
+ *\brief Dump contents of the block to a file descriptor.
+ *
+ *\param fd A writable file descriptor.
+ *\return Always zero.
+ *\throw io_error_exception A write error occured.
+ */
+int tape_block::dump_to_fd(int fd){
+  int to_write=raw_size;
+  unsigned char * dump_ptr=raw_data;
+  int written;
+  while (to_write) {
+    written=write(fd,dump_ptr,to_write);
+    if (written < 0) throw io_error_exception();
+    to_write-=written;
+    dump_ptr+=written;
+  }
+  return 0;
+}
+
+/***************************************************************/
+/*!
+ *\brief Determine if the block marks the end of an object
+ *\retval true The block marks the end of an object.
+ *\retval false The block does not mark the end of an object.
+ *\note This is only a virtual method returning always false.
+ *      Classes which can be the legal last block of an object
+ *      will overwrite this method so that it returns true.
+ */
+bool tape_block::is_endblock(){
+  return false;
+}
+
+/***************************************************************/
+/*!
+ *\brief Get exported symbols.
+ *\return A vector containing any symbols exported in this block.
+ *\note This is only a virtual method returning always an empty
+ *      vector. Classes describing appropriate data blocks
+ *      will override this method.
+ */
+vector<string> tape_block::get_exported_symbols(){
+  vector<string> result;
+  return result;
+}
+
+/***************************************************************/
+/*!
+ *\brief Get called symbols.
+ *\return A vector containing any symbols called in this block.
+ *\note This is only a virtual method returning always an empty
+ *      vector. Classes describing appropriate data blocks
+ *      will override this method.
+ */
+vector<string> tape_block::get_called_symbols(){
+  vector<string> result;
+  return result;
+}
+
+/***************************************************************/
+/*!
+ *\brief Determine if the block has a known type.
+ *\return true if type and subtype are known.
+ *\return false if the block type is not one of the valid end
+ *        nodes of the type tree.
+ */
+bool tape_block::has_known_type(){
+  return m_has_known_type;
+}
+
+/***************************************************************/
+/*!
+ *\brief Static factory method for tape blocks.
  *
  * This method creates a tape_block by using the private
  * constructor with file descriptor support.\n
- * It always returns a valid pointer, errors
- * or exceptional conditions during creation are marked in the
- * object's state.\n
+ *
  * The factory method tries to determine the block's type and then
  * looks for appropriate subclasses. If there's an appropriate
  * subclass fitting the newly-created object, an
@@ -161,7 +241,6 @@ unsigned char * tape_block::get_raw_data(){
  * methods not already present in tape_block.\n
  * Objects of the  following subclasses are automagically generated:\n
  * - eot_block
- * - discard_block
  * - data_block
  *   - data_block_0
  *     - data_block_0_0
@@ -176,6 +255,7 @@ unsigned char * tape_block::get_raw_data(){
  *     - data_block_0_50
  *     - data_block_0_54
  *     - data_block_0_60
+ *     - data_block_0_64
  *   - data_block_1
  *   - data_block_2
  *   - data_block_3
@@ -184,8 +264,8 @@ unsigned char * tape_block::get_raw_data(){
  *   - data_block_6
  *   - data_block_7
  *
- *\return Pointer to an object of type tape_block or a subclass.
- *\param fd A file descriptor where the read is taken from.
+ *\return Pointer to an instance of tape_block or a subclass.
+ *\param fd A file descriptor where data can be read from.
  *\param input_start A pointer to a function 
  *       called before reading data from fd.
  *\param input_stop A pointer to a function
@@ -199,75 +279,82 @@ tape_block * tape_block::gen_from_fd(int  fd,
                                     void (*input_stop)(void *),
                                     void * start_stop_arg)
 {
-  tape_block * res_block;
-  res_block=new tape_block(fd,input_start,input_stop,start_stop_arg);
-  data_block    * d_block;
-  data_block_0  * d0_block;
-  eot_block     * e_block;
-  discard_block * di_block;
-
-  // Retype to data_block if possible
-  switch(res_block->get_type()){
-  case tape_block::TBT_DATA:    // Make pointer a data block
-    d_block=new data_block(*res_block);
-    delete res_block;
-    res_block=d_block;
-    break;
-  case tape_block::TBT_EOT:
-    e_block=new eot_block(*res_block);
-    delete res_block;
-    res_block=e_block;
-    break;
-  case tape_block::TBT_DISCARD:
-    di_block=new discard_block(*res_block);
-    delete res_block;
-    res_block=di_block;
-    break;
-  default:          // All other cases
-    return res_block;
+  tape_block  * n_tape_block=0;
+  data_block  * n_data_block=0;
+  data_block  * n_data_block_x=0;
+  data_block_0  * n_data_block_0=0;
+  data_block_0  * n_data_block_0_x=0;
+  eot_block * n_eot_block=0;
+  broken_block * n_broken_block=0;
+
+  //Use the private constructor which reads in the block from a file descriptor
+  try{
+  n_tape_block=new tape_block(fd,input_start,input_stop,start_stop_arg);
   }
   
-  // Retype again, if it's a data_block now
-  
-  switch(res_block->get_type()){
-  case 00: 
-    d0_block=new data_block_0(*d_block); 
-    d_block=d0_block;
+  catch(checksum_error_exception e){
+    n_broken_block=new broken_block(*(e.get_block()));
+    delete e.get_block();
+    throw checksum_error_exception(n_broken_block);
+  }
+    
+  // Retype to data_block, eot_block - if possible
+  switch(n_tape_block->get_type()){
+  case tape_block::TBT_DATA:
+    n_data_block = new data_block(*n_tape_block); 
     break;
-//   case 01: d_block=new data_block_1(*d_block); break;
-//   case 02: d_block=new data_block_2(*d_block); break;
-//   case 03: d_block=new data_block_3(*d_block); break;
-//   case 04: d_block=new data_block_4(*d_block); break;
-//   case 05: d_block=new data_block_5(*d_block); break;
-//   case 06: d_block=new data_block_6(*d_block); break;
-//   case 07: d_block=new data_block_7(*d_block); break;
+  case tape_block::TBT_EOT:
+    n_eot_block=new eot_block(*n_tape_block); 
+    delete n_tape_block;
+    n_eot_block->m_has_known_type=true;
+    return n_eot_block;
+  default: // Unknown block, a bad thing!         
+    return n_tape_block; 
+  }
+  delete n_tape_block;
+
+  // Now only data blocks are left
+
+  switch(n_data_block->get_type()){
+  case 00: n_data_block_0=new data_block_0(*n_data_block); break;
+  case 01: n_data_block_x=new data_block_1(*n_data_block); break;
+  case 02: n_data_block_x=new data_block_2(*n_data_block); break;
+  case 03: n_data_block_x=new data_block_3(*n_data_block); break;
+  case 04: n_data_block_x=new data_block_4(*n_data_block); break;
+  case 05: n_data_block_x=new data_block_5(*n_data_block); break;
+  case 06: n_data_block_x=new data_block_6(*n_data_block); break;
+  case 07: n_data_block_x=new data_block_7(*n_data_block); break;
   default:
-    return res_block;
+    return n_data_block;
+  }
+  if (n_data_block_0==0){ // So we must have found another one
+    delete n_data_block;
+    n_data_block_x->m_has_known_type=true;
+    return n_data_block_x;
   }
-  delete res_block;
-  res_block=d_block;
   
-  if (res_block->get_type()==0) switch(d0_block->get_subtype()){
-  case 000: d0_block=new data_block_0_0(*d0_block); break;
-  case 001: d0_block=new data_block_0_1(*d0_block); break;
-  case 002: d0_block=new data_block_0_2(*d0_block); break;
-  case 003: d0_block=new data_block_0_3(*d0_block); break;
-  case 004: d0_block=new data_block_0_4(*d0_block); break;
-  case 010: d0_block=new data_block_0_10(*d0_block); break;
-  case 014: d0_block=new data_block_0_14(*d0_block); break;
-  case 024: d0_block=new data_block_0_24(*d0_block); break;
-  case 044: d0_block=new data_block_0_44(*d0_block); break;
-  case 030: d0_block=new data_block_0_30(*d0_block); break;
-  case 054: d0_block=new data_block_0_54(*d0_block); break;
-  case 060: d0_block=new data_block_0_60(*d0_block); break;
-  case 050: d0_block=new data_block_0_50(*d0_block); break;
+  // Here only type 0 left
+  switch(n_data_block_0->get_subtype()){
+  case 000: n_data_block_0_x=new data_block_0_0(*n_data_block_0); break;
+  case 001: n_data_block_0_x=new data_block_0_1(*n_data_block_0); break;
+  case 002: n_data_block_0_x=new data_block_0_2(*n_data_block_0); break;
+  case 003: n_data_block_0_x=new data_block_0_3(*n_data_block_0); break;
+  case 004: n_data_block_0_x=new data_block_0_4(*n_data_block_0); break;
+  case 010: n_data_block_0_x=new data_block_0_10(*n_data_block_0); break;
+  case 014: n_data_block_0_x=new data_block_0_14(*n_data_block_0); break;
+  case 024: n_data_block_0_x=new data_block_0_24(*n_data_block_0); break;
+  case 044: n_data_block_0_x=new data_block_0_44(*n_data_block_0); break;
+  case 030: n_data_block_0_x=new data_block_0_30(*n_data_block_0); break;
+  case 054: n_data_block_0_x=new data_block_0_54(*n_data_block_0); break;
+  case 060: n_data_block_0_x=new data_block_0_60(*n_data_block_0); break;
+  case 064: n_data_block_0_x=new data_block_0_64(*n_data_block_0); break;
+  case 050: n_data_block_0_x=new data_block_0_50(*n_data_block_0); break;
   default:
-    return res_block;
+    return n_data_block_0;
   }
-  delete res_block;
-  res_block=d0_block;
-  return res_block;
+  delete n_data_block_0;
+  n_data_block_0_x->m_has_known_type=true;
+  return n_data_block_0_x;
 }
 
 /***************************************************************/
@@ -295,8 +382,8 @@ string tape_block::get_typestring(){
 /*!
  *\brief Automatic constructor with file descriptor support
  *
- * This constructor is used to initialise the block read in the block's 
- * data via a file descriptor.\n
+ * This constructor is used to initialise the block read in the 
+ * block's data via a file descriptor.\n
  * This is done in the following way:
  * - input_start() is called.
  * - Data is read from fd. Stops on end of file or end of block
@@ -324,30 +411,33 @@ tape_block::tape_block (int  fd,
   int eob_needed=EOB_LENGTH;
   int block_complete=0;
 
-  block_type= TBT_DISCARD;
-  init_state= TBS_DEFAULT;
-  data_read = 0;
+  block_type= TBT_BROKEN;
+  discarded_bytes=0;
   raw_data  = 0;
   raw_size  = 0;
   word_data = 0;
   word_size = 0;
-  
+  poolsize  = 1024;
+
+  // Get initial portion of memory
+  raw_data=(unsigned char*)malloc(poolsize);
+
   if (input_start) input_start (start_stop_arg);
 
   /* Look for block start in input stream */
   do { 
     retval=read(fd, &buffer, 1);
     if (retval == 0){             // End of file
-      block_type=TBT_DISCARD;
-      init_state=TBS_EOF_LEGAL;
+      free(raw_data);
+      throw eof_legal_exception(discarded_bytes);
       return;
     }
     if (retval == -1){            // Error reading from fd
-      block_type=TBT_DISCARD;
-      init_state=TBS_IOERR;
+      free(raw_data);
+      throw io_error_exception();
       return;
     }
-    data_read++;
+    discarded_bytes++;
     
     /* Look for end ot tape sequence */
     switch(eot_needed){
@@ -361,41 +451,48 @@ tape_block::tape_block (int  fd,
       break;
     case 1:
       if (buffer==EOT_SEQ_3){
-       raw_data=(unsigned char*)malloc(3);
+       raw_data=(unsigned char*)realloc(raw_data,3);
        raw_size=3;
+       discarded_bytes-=3;
        raw_data[0]=EOT_SEQ_1;
        raw_data[1]=EOT_SEQ_2;
        raw_data[2]=EOT_SEQ_3;
        block_type =TBT_EOT;
-       init_state =TBS_OK;
        return;
       }
       else eot_needed=EOT_LENGTH;
       break;
     }
   } while (buffer != (unsigned char) BLOCK_START_DELIMITER);
+  discarded_bytes--; // Don't want to declare the start delimiter discarded!
+  raw_size=1;        // But we want to account it here!
   
-
   /* Now input the block data */
   block_complete = 0;
+
+  raw_data[0]=buffer; // Put the start delimiter in!
+
   do {
     retval = read (fd, &buffer, 1);
 
     if (retval == 0){             // End of file
-      block_type=TBT_DISCARD;
-      init_state=TBS_EOF_ILLEGAL;
-      return;
+      broken_block * b_block=new broken_block(*this);
+      free(raw_data);
+      throw eof_illegal_exception(b_block);
     }
     if (retval == -1){            // Error reading from fd
-      block_type=TBT_DISCARD;
-      init_state=TBS_IOERR;
-      return;
+      free(raw_data);
+      throw io_error_exception();
     }
-    data_read++;                  // We have consumed a byte
     
     /* sort in the new byte */
     raw_size++;
-    raw_data=(unsigned char*)realloc(raw_data,raw_size);
+    // Need more memory??
+    if (raw_size>poolsize){
+      poolsize+=1024;
+      raw_data=(unsigned char*)realloc(raw_data,poolsize);
+    }
+
     raw_data[raw_size-1]=buffer;
 
     /* Look for end ot block sequence */
@@ -406,30 +503,32 @@ tape_block::tape_block (int  fd,
       break;
     case 1:
       if (buffer==BLOCK_END_DELIMITER_2){
-       raw_size-=EOB_LENGTH; // Don't want the delimiter(s) in the data!
-       raw_data=(unsigned char*)realloc(raw_data,raw_size);
        block_complete=1;
       }
       else eob_needed=EOB_LENGTH;
       break;
     }
   }while (!block_complete);
+
   if (input_stop) input_stop (start_stop_arg);
   
+
   /* Now we have the data in. */
   
   if (init_words()!=0){
-    init_state=TBS_CHECKSUM;
-    return;
+    broken_block * b_block=new broken_block(*this);
+    free(raw_data);
+    free(word_data);
+    throw checksum_error_exception(b_block);
   }
-  
   if (test_checksum()!=0){
-    init_state=TBS_CHECKSUM;
-    return;
+    broken_block * b_block=new broken_block(*this);
+    free(raw_data);
+    free(word_data);
+        throw checksum_error_exception(b_block);
   }
-  
   block_type=TBT_DATA;    
-  init_state=TBS_OK;
 }
 
 
@@ -440,11 +539,11 @@ tape_block::tape_block (int  fd,
  *\retval 1 on error.
  */
 int tape_block::init_words(){
-  word_size=raw_size/3;
+  word_size=(raw_size-EOB_LENGTH-1)/3;
   if (word_size<MINIMUM_DATA_BLOCK_WORDSIZE) return 1; // Block too short!
-  word_data=(unsigned short *)malloc(raw_size*sizeof(unsigned short));
+  word_data=(unsigned short *)malloc(word_size*sizeof(unsigned short));
   for (int pos=0; pos<word_size; pos++)
-    word_data[pos]=combine466(raw_data+3*pos);
+    word_data[pos]=combine466(raw_data+1+3*pos);
   return 0;
 }
 
@@ -462,3 +561,82 @@ int tape_block::test_checksum(){
   else return 1;
 }
 
+/***************************************************************/
+/*!
+ *\brief Standard constructor of exception base class.
+ */
+tape_block::exception::exception(){
+}
+
+/***************************************************************/
+/*!
+ *\brief Standard desctructor.
+ */
+tape_block::exception::~exception(){
+}
+
+/***************************************************************/
+/*!
+ *\brief Get pointer to erroneous tape_block
+ *\return A pointer to a tape_block instance.
+ *\note The user has to take care to free the instance contained
+ *      in objects of classes supporting this method.
+ */
+tape_block * tape_block::exception::get_block(){
+  return m_broken_block;
+}
+
+/***************************************************************/
+/*!
+ *\brief Constructor
+ */
+tape_block::checksum_error_exception::
+checksum_error_exception (tape_block * broken_block){
+  this->m_broken_block=broken_block;
+}
+
+/***************************************************************/
+/*!
+ *\brief Constructor
+ */
+tape_block::eof_illegal_exception::
+eof_illegal_exception (tape_block * broken_block){
+  this->m_broken_block=broken_block;
+}
+
+/***************************************************************/
+/*!
+ *\brief Constructor
+ *\param bytes_consumed Amount of bytes read while looking
+ *       for a block.
+ */
+tape_block::eof_legal_exception::
+eof_legal_exception(int bytes_consumed){
+  this->m_broken_block=NULL;
+  this->bytes_consumed=bytes_consumed;
+}
+
+/***************************************************************/
+/*!
+ *\brief Get amount of consumed data before EOF came along.
+ *\return Amound of bytes read from fd while looking for
+ *        new block or EOF.
+ */
+int tape_block::eof_legal_exception::get_consumed(){
+  return bytes_consumed;
+}
+
+/***************************************************************/
+/*!
+ *\brief Constructor
+ */
+tape_block::io_error_exception::
+io_error_exception(){
+  this->m_broken_block=NULL;
+}
+
+
+
+
+
+