/******************************************************************************
*
*  NSSDC/CDF                                CDF library allocation functions.
*
*  Version 1.0, 22-Jul-96, Hughes STX.
*
*  Modification history:
*
*   V1.0  22-Jul-96, J Love     Original version.
*   V2.0  29-Jun-04, M Liu      Added LFS (Large File Support > 2G).
*
******************************************************************************/

#include "cdflib.h"
#include "cdflib64.h"
#include "cdfrev.h"

/******************************************************************************
* Local function prototypes.
******************************************************************************/

static CDFstatus InsertRecords PROTOARGs((
  struct CDFstruct *CDF, struct VarStruct *Var, struct AllocStruct alloc,
  OFF_T vxrStart, int *count, Int32 *toRec
));
static CDFstatus AppendRecords PROTOARGs((
  struct CDFstruct *CDF, struct VarStruct *Var, struct AllocStruct *alloc,
  OFF_T vxrStart, int *count
));
static CDFstatus InsertEntry PROTOARGs((
  struct CDFstruct *CDF, struct VarStruct *Var, struct AllocStruct *alloc,
  OFF_T vxrOffset, int atEntryN, int *count
));
static void InsertAtEntry PROTOARGs((
  Int32 first, Int32 last, OFF_T offset, struct VXRstruct64 *VXR, int atEntryN, 
  Logical *push, Int32 *pushFirst, Int32 *pushLast, OFF_T *pushOffset
));
static CDFstatus AppendEntry PROTOARGs((
  struct CDFstruct *CDF, struct VarStruct *Var, struct AllocStruct *alloc,
  OFF_T vxrOffset, int *count
));
static CDFstatus PadSparseRecords PROTOARGs((
  struct CDFstruct *CDF, struct VarStruct *Var, OFF_T offset, Int32 nRecords
));
static CDFstatus CountVXRs PROTOARGs((
  struct CDFstruct *CDF, OFF_T vxrStart, int *count
));
static void InitNewVXRx PROTOARGs((struct VXRstruct64 *VXRx));
static CDFstatus FirstRecords PROTOARGs((
  struct CDFstruct *CDF, struct VarStruct *Var, struct AllocStruct *alloc
));
static CDFstatus ExtendLevel PROTOARGs((
  struct CDFstruct *CDF, OFF_T vxrOffset, int count, OFF_T *vxrOffsetHead,
  Int32 *lastNew, int *countNew
));
static CDFstatus CreateLevel PROTOARGs((
  struct CDFstruct *CDF, OFF_T vxrOffset, int count, OFF_T *vxrOffsetHead,
  int *newCount
));
static CDFstatus AllocateVR PROTOARGs((
  struct CDFstruct *CDF, struct VarStruct *Var, struct AllocStruct *alloc,
  OFF_T *offset
));
static CDFstatus AllocateVVR PROTOARGs((
  struct CDFstruct *CDF, struct VarStruct *Var, Int32 nRecords,
  OFF_T *vvrOffset
));
static CDFstatus AllocateCVVR PROTOARGs((
  struct CDFstruct *CDF, OFF_T cSize, OFF_T xSize, OFF_T *cvvrOffset
));

/******************************************************************************
* AllocateRecords64.
******************************************************************************/

STATICforIDL CDFstatus AllocateRecords64 (CDF, Var, alloc)
struct CDFstruct *CDF;
struct VarStruct *Var;
struct AllocStruct alloc;
{
  CDFstatus pStatus = CDF_OK;
  Int32 toRec; int count, newCount;
  OFF_T vxrHead, vxrHeadNew;
  /****************************************************************************
  * Read the head of the VXR tree/list.
  ****************************************************************************/
  if (!sX(ReadVDR64(CDF,CDF->fp,Var->VDRoffset64,Var->zVar,
		    VDR_VXRHEAD,&vxrHead,
		    VDR_NULL),&pStatus)) return pStatus;
  /****************************************************************************
  * If no records allocated yet...
  ****************************************************************************/
  if (vxrHead == (OFF_T) ZERO_OFFSET64) {
    if (!sX(FirstRecords(CDF,Var,&alloc),&pStatus)) return pStatus;
    return pStatus;
  }
  /****************************************************************************
  * Allocate records starting at the VXRhead until done.
  ****************************************************************************/
  while (alloc.first <= alloc.last) {
    /**************************************************************************
    * Try to insert all of the records.
    **************************************************************************/
    if (!sX(InsertRecords(CDF,Var,
			  alloc,vxrHead,
			  &count,&toRec),&pStatus)) return pStatus;
    /**************************************************************************
    * Check to see if another level of indexing needs to be added.  Multiple
    * levels may be created here because prior to CDF V2.6 a hierarchical
    * indexing scheme was not used.
    **************************************************************************/
    while (count > NUM_VXRx_ENTRIES - 1) {
      if (!sX(CreateLevel(CDF,vxrHead,count,
			  &vxrHeadNew,&newCount),&pStatus)) return pStatus;
      vxrHead = vxrHeadNew;
      if (!sX(WriteVDR64(CDF,CDF->fp,Var->VDRoffset64,Var->zVar,
		         VDR_VXRHEAD,&vxrHead,
		         VDR_NULL),&pStatus)) return pStatus;
      count = newCount;
    }
    /**************************************************************************
    * Adjust the allocation request based on the record actually allocated to.
    **************************************************************************/
    alloc.first = toRec + 1;
  }
  /****************************************************************************
  * Update the VXR tail in case it changed.  This assumes that the VXR tail
  * field in the VDR is not used by any of the record allocation routines.
  ****************************************************************************/
  if (!sX(UpdateVXRtailInVDR64(CDF,Var),&pStatus)) return pStatus;
  return pStatus;
}

/******************************************************************************
* InsertRecords.
******************************************************************************/

static CDFstatus InsertRecords (CDF, Var, alloc, vxrStart, count, toRec)
struct CDFstruct *CDF;
struct VarStruct *Var;
struct AllocStruct alloc;
OFF_T vxrStart;
int *count;
Int32 *toRec;
{
  CDFstatus pStatus = CDF_OK; struct VXRstruct64 VXR;
  OFF_T vxrOffset = vxrStart, vxrOffsetNew;
  Int32 irType, lastNew; 
  int entryN, belowCount, belowCountNew, toEndCount, countNew;
  /****************************************************************************
  * Scan linked list of VXRs...
  ****************************************************************************/
  *count = 0;
  while (vxrOffset != (OFF_T) ZERO_OFFSET64) {
    /**************************************************************************
    * Read VXR.
    **************************************************************************/
    if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		      VXR_RECORD,&VXR,
		      VXR_NULL),&pStatus)) return pStatus;
    (*count)++;
    /**************************************************************************
    * Scan VXR entries...
    **************************************************************************/
    for (entryN = 0; entryN < VXR.NusedEntries; entryN++) {
       if (alloc.first < VXR.First[entryN]) {
	 if (!sX(ReadIrType64(CDF->fp,
			      VXR.Offset[entryN],
			      &irType),&pStatus)) return pStatus;
	 switch (irType) {
	   case VXR_: {
	     Int32 prevRecN = VXR.First[entryN] - 1;
	     alloc.last = MINIMUM(alloc.last,prevRecN);
	     if (!sX(InsertRecords(CDF,Var,alloc,
				   VXR.Offset[entryN],
				   &belowCount,
				   toRec),&pStatus)) return pStatus;
	     VXR.First[entryN] = alloc.first;
	     while (belowCount > NUM_VXRx_ENTRIES - 1) {
	       if (!sX(CreateLevel(CDF,VXR.Offset[entryN],
				   belowCount,&VXR.Offset[entryN],
				   &belowCountNew),&pStatus)) return pStatus;
	       belowCount = belowCountNew;
	     }
	     if (!sX(WriteVXR64(CDF->fp,vxrOffset,
			        VXR_RECORD,&VXR,
			        VXR_NULL),&pStatus)) return pStatus;
	     if (!sX(CountVXRs(CDF,VXR.VXRnext,
			       &toEndCount),&pStatus)) return pStatus;
	     *count += toEndCount;
	     return pStatus;
	   }
	   case VVR_:
	   case CVVR_:
	     if (alloc.last > VXR.First[entryN] - 1) {
	       if (EXISTSisBAD(&alloc)) return CDF_INTERNAL_ERROR;
	       alloc.last = VXR.First[entryN] - 1;
	       sX (SOME_ALREADY_ALLOCATED, &pStatus);
	     }
	     if (!sX(InsertEntry(CDF,Var,
				 &alloc,vxrOffset,
				 entryN,&toEndCount),&pStatus)) return pStatus;
	     *count += (toEndCount - 1);
	     *toRec = alloc.last;
	     return pStatus;
	   default:
	     return CORRUPTED_V3_CDF;
	 }
       }
       else {
	 if (alloc.first <= VXR.Last[entryN]) {
	   if (!sX(ReadIrType64(CDF->fp,
			        VXR.Offset[entryN],
			        &irType),&pStatus)) return pStatus;
	   switch (irType) {
	     case VXR_:
	       alloc.last = MINIMUM(alloc.last,VXR.Last[entryN]);
	       if (!sX(InsertRecords(CDF,Var,alloc,
				     VXR.Offset[entryN],
				     &belowCount,
				     toRec),&pStatus)) return pStatus;
	       while (belowCount > NUM_VXRx_ENTRIES - 1) {
		 if (!sX(CreateLevel(CDF,VXR.Offset[entryN],
				     belowCount,&VXR.Offset[entryN],
				     &belowCountNew),&pStatus)) return pStatus;
		 if (!sX(WriteVXR64(CDF->fp,vxrOffset,
				    VXR_RECORD,&VXR,
				    VXR_NULL),&pStatus)) return pStatus;
		 belowCount = belowCountNew;
	       }
	       if (!sX(CountVXRs(CDF,VXR.VXRnext,
				 &toEndCount),&pStatus)) return pStatus;
	       *count += toEndCount;
	       return pStatus;
	     case VVR_:
	     case CVVR_:
	       if (EXISTSisBAD(&alloc)) return CDF_INTERNAL_ERROR;
	       sX (SOME_ALREADY_ALLOCATED, &pStatus);
	       *toRec = MINIMUM(VXR.Last[entryN],alloc.last);
	       if (!sX(CountVXRs(CDF,
				 VXR.VXRnext,
				 &toEndCount),&pStatus)) return pStatus;
	       *count += toEndCount;
	       return pStatus;
	     default:
	       return CORRUPTED_V3_CDF;
	   }
	 }
       }
    }
    /**************************************************************************
    * No more entries for this VXR.  If there are no more VXRs...
    **************************************************************************/
    if (VXR.VXRnext == (OFF_T) ZERO_OFFSET64) {
      int lastEntryN = entryN - 1;
      if (!sX(ReadIrType64(CDF->fp,
			   VXR.Offset[lastEntryN],
			   &irType),&pStatus)) return pStatus;
      switch (irType) {
	case VXR_:
	  if (!sX(AppendRecords(CDF,Var,&alloc,
				VXR.Offset[lastEntryN],
				&belowCount),&pStatus)) return pStatus;
	  VXR.Last[lastEntryN] = alloc.last;
	  if (belowCount > NUM_VXRx_ENTRIES) {
	    if (!sX(ExtendLevel(CDF,VXR.Offset[lastEntryN],
				belowCount,&vxrOffsetNew,
				&lastNew,&countNew),&pStatus)) return pStatus;
	    VXR.Last[lastEntryN] = lastNew;
	    VXR.VXRnext = vxrOffsetNew;
	    *count += countNew;
	  }
	  if (!sX(WriteVXR64(CDF->fp,vxrOffset,
			     VXR_RECORD,&VXR,
			     VXR_NULL),&pStatus)) return pStatus;
	  *toRec = alloc.last;
	  break;
	case VVR_:
	case CVVR_:
	  if (!sX(AppendEntry(CDF,Var,&alloc,
			      vxrOffset,&toEndCount),&pStatus)) return pStatus;
	  *count += (toEndCount - 1);
	  *toRec = alloc.last;
	  break;
	default:
	  return CORRUPTED_V3_CDF;
      }
      return pStatus;
    }
    /**************************************************************************
    * Go on to the next VXR.
    **************************************************************************/
    vxrOffset = VXR.VXRnext;
  }
  return pStatus;
}

/******************************************************************************
* AppendRecords.
******************************************************************************/

static CDFstatus AppendRecords (CDF, Var, alloc, vxrStart, count)
struct CDFstruct *CDF;
struct VarStruct *Var;
struct AllocStruct *alloc;
OFF_T vxrStart;
int *count;
{
  CDFstatus pStatus = CDF_OK; struct VXRstruct64 VXR;
  OFF_T vxrOffset = vxrStart, vxrOffsetNew;
  Int32 irType, lastNew;
  int lastEntryN, belowCount, toEndCount, countNew;
  /****************************************************************************
  * Locate last VXR on linked list.  We had better be appending at the last
  * VXR.
  ****************************************************************************/
  if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		    VXR_RECORD,&VXR,
		    VXR_NULL),&pStatus)) return pStatus;
  *count = 1;
  while (VXR.VXRnext != (OFF_T) ZERO_OFFSET64) {
    vxrOffset = VXR.VXRnext;
    if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		      VXR_RECORD,&VXR,
		      VXR_NULL),&pStatus)) return pStatus;
    (*count)++;
  }
  /****************************************************************************
  * If the last entry points to another VXR, recursively call this routine
  * with the offset of that VXR.  Otherwise, append the entry.
  ****************************************************************************/
  lastEntryN = (int) (VXR.NusedEntries - 1);
  if (!sX(ReadIrType64(CDF->fp,
		       VXR.Offset[lastEntryN],
		       &irType),&pStatus)) return pStatus;
  switch (irType) {
    case VXR_:
      if (!sX(AppendRecords(CDF,Var,alloc,
			    VXR.Offset[lastEntryN],
			    &belowCount),&pStatus)) return pStatus;
      VXR.Last[lastEntryN] = alloc->last;
      if (belowCount > NUM_VXRx_ENTRIES) {
	if (!sX(ExtendLevel(CDF,VXR.Offset[lastEntryN],
			    belowCount,&vxrOffsetNew,
			    &lastNew,&countNew),&pStatus)) return pStatus;
	VXR.Last[lastEntryN] = lastNew;
	VXR.VXRnext = vxrOffsetNew;
	*count += countNew;
      }
      if (!sX(WriteVXR64(CDF->fp,vxrOffset,
		         VXR_RECORD,&VXR,
		         VXR_NULL),&pStatus)) return pStatus;
      break;
    case VVR_:
    case CVVR_:
      if (!sX(AppendEntry(CDF,Var,alloc,
			  vxrOffset,&toEndCount),&pStatus)) return pStatus;
      *count += (toEndCount - 1);
      break;
    default:
      return CORRUPTED_V3_CDF;
  }
  return pStatus;
}

/******************************************************************************
* InsertEntry.
******************************************************************************/

static CDFstatus InsertEntry (CDF, Var, alloc, vxrOffset, atEntryN, count)
struct CDFstruct *CDF;
struct VarStruct *Var;
struct AllocStruct *alloc;
OFF_T vxrOffset;
int atEntryN;
int *count;
{
  CDFstatus pStatus = CDF_OK;
  OFF_T vxrOffsetNew, vrOffset, pushOffset; Int32 pushFirst, pushLast;
  struct VXRstruct64 VXR; Logical push;
  /****************************************************************************
  * Read the VXR, insert the entry while determining if an entry was pushed
  * out of the VXR, and rewrite the VXR.
  ****************************************************************************/
  if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		    VXR_RECORD,&VXR,
		    VXR_NULL),&pStatus)) return pStatus;
  if (!sX(AllocateVR(CDF,Var,alloc,&vrOffset),&pStatus)) return pStatus;
  InsertAtEntry (alloc->first, alloc->last, vrOffset, &VXR, atEntryN,
		 &push, &pushFirst, &pushLast, &pushOffset);
  if (!sX(WriteVXR64(CDF->fp,vxrOffset,
		     VXR_RECORD,&VXR,
		     VXR_NULL),&pStatus)) return pStatus;
  *count = 1;
  /****************************************************************************
  * While an entry was pushed out...
  ****************************************************************************/
  while (push) {
    if (VXR.VXRnext != (OFF_T) ZERO_OFFSET64) {
      vxrOffset = VXR.VXRnext;
      if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		        VXR_RECORD,&VXR,
		        VXR_NULL),&pStatus)) return pStatus;
      InsertAtEntry (pushFirst, pushLast, pushOffset, &VXR, 0, &push,
		     &pushFirst, &pushLast, &pushOffset);
      if (!sX(WriteVXR64(CDF->fp,vxrOffset,
		         VXR_RECORD,&VXR,
		         VXR_NULL),&pStatus)) return pStatus;
    }
    else {
      if (!sX(AllocateIR64(CDF,((Int32)VXR_BASE_SIZE64),
			   &vxrOffsetNew),&pStatus)) return pStatus;
      if (!sX(WriteVXR64(CDF->fp,vxrOffset,
		         VXR_VXRNEXT,&vxrOffsetNew,
		         VXR_NULL),&pStatus)) return pStatus;
      InitNewVXR64 (&VXR, pushFirst, pushLast, pushOffset);
      if (!sX(WriteVXR64(CDF->fp,vxrOffsetNew,
		         VXR_RECORD,&VXR,
		         VXR_NULL),&pStatus)) return pStatus;
      push = FALSE;
    }
    (*count)++;
  }
  return pStatus;
}

/******************************************************************************
* InsertAtEntry.
******************************************************************************/

static void InsertAtEntry (first, last, offset, VXR, atEntryN, push,
			     pushFirst, pushLast, pushOffset)
Int32 first;
Int32 last;
OFF_T offset;
struct VXRstruct64 *VXR;
int atEntryN;
Logical *push;
Int32 *pushFirst;
Int32 *pushLast;
OFF_T *pushOffset;
{
  int startEntryN, entryN;
  if (VXR->NusedEntries < VXR->Nentries) {
    *push = FALSE;
    startEntryN = (int) VXR->NusedEntries;
    VXR->NusedEntries++;
  }
  else {
    startEntryN = (int) (VXR->Nentries - 1);
    *pushFirst = VXR->First[startEntryN];
    *pushLast = VXR->Last[startEntryN];
    *pushOffset = VXR->Offset[startEntryN];
    *push = TRUE;
  }
  for (entryN = startEntryN; entryN > atEntryN; entryN--) {
     VXR->First[entryN] = VXR->First[entryN-1];
     VXR->Last[entryN] = VXR->Last[entryN-1];
     VXR->Offset[entryN] = VXR->Offset[entryN-1];
  }
  VXR->First[atEntryN] = first;
  VXR->Last[atEntryN] = last;
  VXR->Offset[atEntryN] = offset;
  return;
}

/******************************************************************************
* AppendEntry.
*   It is assumed that the offset is for a VXR at the end of a linked list.
******************************************************************************/

static CDFstatus AppendEntry (CDF, Var, alloc, vxrOffset, count)
struct CDFstruct *CDF;
struct VarStruct *Var;
struct AllocStruct *alloc;
OFF_T vxrOffset;
int *count;
{
  CDFstatus pStatus = CDF_OK; Logical resized;
  struct VXRstruct64 VXR, newVXR; OFF_T offset, vxrOffsetNew;
  /****************************************************************************
  * Read the VXR.
  ****************************************************************************/
  if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		    VXR_RECORD,&VXR,
		    VXR_NULL),&pStatus)) return pStatus;
  /****************************************************************************
  * First check if a VVR_ can be extended.
  ****************************************************************************/
  if (alloc->type == VVR_) {
    if (!alloc->vvr.new) {
      int lastEntryN = (int) (VXR.NusedEntries - 1);
      if (alloc->first == VXR.Last[lastEntryN] + 1) {
	Int32 newCount = alloc->last - VXR.First[lastEntryN] + 1;
	OFF_T newSize = (OFF_T) (VVR_BASE_SIZE64 + 
				 ((OFF_T)newCount) * Var->NphyRecBytes);
	if (!sX(ResizeIR64(CDF,VXR.Offset[lastEntryN],
			   newSize,NULL,
			   FALSE,&resized),&pStatus)) return pStatus;
	if (resized) {
	  VXR.Last[lastEntryN] = alloc->last;
	  if (!sX(WriteVXR64(CDF->fp,vxrOffset,
			     VXR_RECORD,&VXR,
			     VXR_NULL),&pStatus)) return pStatus;
	  if (!sX(WriteVVR64(CDF->fp,VXR.Offset[lastEntryN],
			     VVR_RECORDSIZE,&newSize,
			     VVR_NULL),&pStatus)) return pStatus;
	  switch (Var->vType) {
	    case SPARSE_RECORDS_:
	      if (alloc->last < Var->maxWritten) {
		Int32 nRecords = alloc->first - VXR.First[lastEntryN];
		offset = VXR.Offset[lastEntryN] +
			 VVR_BASE_SIZE64 +
			 (nRecords * Var->NphyRecBytes);
		if (!sX(PadSparseRecords(CDF,Var,
					 offset,
					 nRecords),&pStatus)) return pStatus;
	      }
	      break;
	  }
	  *count = 1;
	  return pStatus;
	}
      }
    }
  }
  /****************************************************************************
  * If there is an unused entry...
  ****************************************************************************/
  if (VXR.NusedEntries < VXR.Nentries) {
    int entryN = (int) VXR.NusedEntries;
    VXR.First[entryN] = alloc->first;
    VXR.Last[entryN] = alloc->last;
    if (!sX(AllocateVR(CDF,Var,alloc,
		       &VXR.Offset[entryN]),&pStatus)) return pStatus;
    VXR.NusedEntries++;
    if (!sX(WriteVXR64(CDF->fp,vxrOffset,
		       VXR_RECORD,&VXR,
		       VXR_NULL),&pStatus)) return pStatus;
    *count = 1;
    return pStatus;
  }
  /****************************************************************************
  * No unused entries - create a new VXR.
  ****************************************************************************/
  if (!sX(AllocateIR64(CDF,((Int32)VXR_BASE_SIZE64),
		       &vxrOffsetNew),&pStatus)) return pStatus;
  if (!sX(AllocateVR(CDF,Var,alloc,&offset),&pStatus)) return pStatus;
  if (!sX(WriteVXR64(CDF->fp,vxrOffset,
		     VXR_VXRNEXT,&vxrOffsetNew,
		     VXR_NULL),&pStatus)) return pStatus;
  InitNewVXR64 (&newVXR, alloc->first, alloc->last, offset);
  if (!sX(WriteVXR64(CDF->fp,vxrOffsetNew,
		     VXR_RECORD,&newVXR,
		     VXR_NULL),&pStatus)) return pStatus;
  *count = 2;
  return pStatus;
}

/******************************************************************************
* PadSparseRecords.
******************************************************************************/

static CDFstatus PadSparseRecords (CDF, Var, offset, nRecords)
struct CDFstruct *CDF;
struct VarStruct *Var;
OFF_T offset;
Int32 nRecords;
{
  CDFstatus pStatus = CDF_OK; int how; void *buffer;
  if (!sX(BuildPadBuffer64(CDF,Var,nRecords,
			   &how,&buffer,TRUE),&pStatus)) return pStatus;
  if (!sX(WritePadValues64(Var,CDF->fp,offset,
			   nRecords,how,buffer),&pStatus)) {
    cdf_FreeMemory (buffer, NULL);
    return pStatus;
  }
  cdf_FreeMemory (buffer, NULL);
  return pStatus;
}

/******************************************************************************
* CountVXRs.
******************************************************************************/

static CDFstatus CountVXRs (CDF, vxrStart, count)
struct CDFstruct *CDF;
OFF_T vxrStart;
int *count;
{
  CDFstatus pStatus = CDF_OK; OFF_T vxrOffset = vxrStart;
  *count = 0;
  while (vxrOffset != (OFF_T) ZERO_OFFSET64) {
    if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		      VXR_VXRNEXT,&vxrOffset,
		      VXR_NULL),&pStatus)) return pStatus;
    (*count)++;
  }
  return pStatus;
}

/******************************************************************************
* InitNewVXR64.
******************************************************************************/

STATICforIDL void InitNewVXR64 (VXR, firstRec, lastRec, offset)
struct VXRstruct64 *VXR;
Int32 firstRec;
Int32 lastRec;
OFF_T offset;
{
  int entryN;
  VXR->RecordSize = VXR_BASE_SIZE64;
  VXR->RecordType = VXR_;
  VXR->VXRnext = (OFF_T) ZERO_OFFSET64;
  VXR->Nentries = NUM_VXR_ENTRIES;
  VXR->NusedEntries = 1;
  VXR->First[0] = firstRec;
  VXR->Last[0] = lastRec;
  VXR->Offset[0] = offset;
  for (entryN = 1; entryN < NUM_VXR_ENTRIES; entryN++) {
     VXR->First[entryN] = NO_RECORD;
     VXR->Last[entryN] = NO_RECORD;
     VXR->Offset[entryN] = (OFF_T) NO_OFFSET64;
  }
  return;
}

/******************************************************************************
* InitNewVXRx.
******************************************************************************/

static void InitNewVXRx (VXRx)
struct VXRstruct64 *VXRx;
{
  int entryN;
  VXRx->RecordSize = VXRx_BASE_SIZE64;
  VXRx->RecordType = VXR_;
  VXRx->VXRnext = (OFF_T) 0;
  VXRx->Nentries = NUM_VXRx_ENTRIES;
  VXRx->NusedEntries = 0;
  for (entryN = 0; entryN < NUM_VXRx_ENTRIES; entryN++) {
     VXRx->First[entryN] = NO_RECORD;
     VXRx->Last[entryN] = NO_RECORD;
     VXRx->Offset[entryN] = (OFF_T) NO_OFFSET64;
  }
  return;
}

/******************************************************************************
* FirstRecords.
******************************************************************************/

static CDFstatus FirstRecords (CDF, Var, alloc)
struct CDFstruct *CDF;
struct VarStruct *Var;
struct AllocStruct *alloc;
{
  CDFstatus pStatus = CDF_OK;
  struct VXRstruct64 VXR; OFF_T offset, vxrOffset;
  if (!sX(AllocateIR64(CDF,VXR_BASE_SIZE64,&vxrOffset),&pStatus)) 
	return pStatus;
  if (!sX(AllocateVR(CDF,Var,alloc,&offset),&pStatus)) return pStatus;
  InitNewVXR64 (&VXR, alloc->first, alloc->last, offset);
  if (!sX(WriteVXR64(CDF->fp,vxrOffset,
		     VXR_RECORD,&VXR,
		     VXR_NULL),&pStatus)) return pStatus;
  if (!sX(WriteVDR64(CDF,CDF->fp,Var->VDRoffset64,Var->zVar,
		     VDR_VXRHEAD,&vxrOffset,
		     VDR_VXRTAIL,&vxrOffset,
		     VDR_NULL),&pStatus)) return pStatus;
  return pStatus;
}

/******************************************************************************
* ExtendLevel.
******************************************************************************/

static CDFstatus ExtendLevel (CDF, vxrOffset, count, vxrOffsetHead, lastNew,
			        countNew)
struct CDFstruct *CDF;
OFF_T vxrOffset;
int count;
OFF_T *vxrOffsetHead;
Int32 *lastNew;
int *countNew;
{
  CDFstatus pStatus = CDF_OK; struct VXRstruct64 VXR;
  if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		    VXR_RECORD,&VXR,
		    VXR_NULL),&pStatus)) return pStatus;
  *lastNew = VXR.Last[(int)(VXR.NusedEntries-1)];
  if (!sX(CreateLevel(CDF,VXR.VXRnext,count-1,
		      vxrOffsetHead,countNew),&pStatus)) return pStatus;
  VXR.VXRnext = (OFF_T) ZERO_OFFSET64;
  if (!sX(WriteVXR64(CDF->fp,vxrOffset,
		     VXR_RECORD,&VXR,
		     VXR_NULL),&pStatus)) return pStatus;
  return pStatus;
}

/******************************************************************************
* CreateLevel.
******************************************************************************/

static CDFstatus CreateLevel (CDF, vxrOffset, count, vxrOffsetHead, newCount)
struct CDFstruct *CDF;
OFF_T vxrOffset;
int count;
OFF_T *vxrOffsetHead;
int *newCount;
{
  CDFstatus pStatus = CDF_OK;
  Int32 sizeOfVXRx = VXRx_BASE_SIZE64;
  OFF_T vxrOffsetNext, vxrOffsetNew;
  int remaining = count, entryN;
  struct VXRstruct64 VXR, newVXR;
  /****************************************************************************
  * Create/initialize first VXRx_.
  ****************************************************************************/
  if (!sX(AllocateIR64(CDF,sizeOfVXRx,vxrOffsetHead),&pStatus)) return pStatus;
  InitNewVXRx (&newVXR);
  *newCount = 1;
  vxrOffsetNew = *vxrOffsetHead;
  /****************************************************************************
  * Until no more VXRs...
  ****************************************************************************/
  while (vxrOffset != (OFF_T) ZERO_OFFSET64) {
    /**************************************************************************
    * Read VXR.
    **************************************************************************/
    if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		      VXR_RECORD,&VXR,
		      VXR_NULL),&pStatus)) return pStatus;
    /**************************************************************************
    * If the current VXRx_ is full...
    **************************************************************************/
    if (newVXR.NusedEntries == newVXR.Nentries) {
      if (!sX(AllocateIR64(CDF,sizeOfVXRx,
			   &(newVXR.VXRnext)),&pStatus)) return pStatus;
      if (!sX(WriteVXR64(CDF->fp,vxrOffsetNew,
		         VXR_RECORD,&newVXR,
		         VXR_NULL),&pStatus)) return pStatus;
      vxrOffsetNew = newVXR.VXRnext;
      InitNewVXRx (&newVXR);
      (*newCount)++;
    }
    /**************************************************************************
    * Determine entry to be used and increment number of used entries.
    **************************************************************************/
    entryN = (int) newVXR.NusedEntries;
    newVXR.NusedEntries++;
    /**************************************************************************
    * If this is the last entry in the VXRx_ and there aren't enough VXRs for
    * another full VXRx_...
    **************************************************************************/
    if (entryN == newVXR.Nentries - 1) {
      if (remaining < NUM_VXRx_ENTRIES + 1) {
	newVXR.First[entryN] = VXR.First[0];
	while (VXR.VXRnext != (OFF_T) ZERO_OFFSET64) {
	  if (!sX(ReadVXR64(CDF->fp,VXR.VXRnext,
			    VXR_RECORD,&VXR,
			    VXR_NULL),&pStatus)) return pStatus;
	}
	newVXR.Last[entryN] = VXR.Last[(int)(VXR.NusedEntries-1)];
	newVXR.Offset[entryN] = vxrOffset;
	break;
      }
    }
    /**************************************************************************
    * Add this VXR's indexing to the current VXRx_.
    **************************************************************************/
    newVXR.First[entryN] = VXR.First[0];
    newVXR.Last[entryN] = VXR.Last[(int)(VXR.NusedEntries-1)];
    newVXR.Offset[entryN] = vxrOffset;
    /**************************************************************************
    * Break the VXR's link to the next VXR.
    **************************************************************************/
    vxrOffsetNext = VXR.VXRnext;
    VXR.VXRnext = (OFF_T) ZERO_OFFSET64;
    if (!sX(WriteVXR64(CDF->fp,vxrOffset,
		       VXR_RECORD,&VXR,
		       VXR_NULL),&pStatus)) return pStatus;
    vxrOffset = vxrOffsetNext;
    /**************************************************************************
    * Decrement the number of remaining VXRs.
    **************************************************************************/
    remaining--;
  }
  /****************************************************************************
  * Write the VXRx_ before returning.
  ****************************************************************************/
  if (!sX(WriteVXR64(CDF->fp,vxrOffsetNew,
		     VXR_RECORD,&newVXR,
		     VXR_NULL),&pStatus)) return pStatus;
  return pStatus;
}

/******************************************************************************
* AllocateVR.
******************************************************************************/

static CDFstatus AllocateVR (CDF, Var, alloc, offset)
struct CDFstruct *CDF;
struct VarStruct *Var;
struct AllocStruct *alloc;
OFF_T *offset;
{
  CDFstatus pStatus = CDF_OK;
  switch (alloc->type) {
    case VVR_: {
      Int32 nRecords = alloc->last - alloc->first + 1;
      if (!sX(AllocateVVR(CDF,Var,
			  nRecords,
			  offset),&pStatus)) return pStatus;
      switch (Var->vType) {
	case SPARSE_RECORDS_:
	  if (alloc->last < Var->maxWritten) {
	    OFF_T tOffset = *offset + VVR_BASE_SIZE64;
	    if (!sX(PadSparseRecords(CDF,Var,
				     tOffset,
				     nRecords),&pStatus)) return pStatus;
	  }
	  break;
      }
      break;
    }
    case CVVR_:
      if (!sX(AllocateCVVR(CDF,alloc->cvvr64.cSize,
			   alloc->cvvr64.xSize,
			   offset),&pStatus)) return pStatus;
      break;
    default:
      return CDF_INTERNAL_ERROR;
  }
  return pStatus;
}

/******************************************************************************
* AllocateVVR.
******************************************************************************/

static CDFstatus AllocateVVR (CDF, Var, nRecords, vvrOffset)
struct CDFstruct *CDF;
struct VarStruct *Var;
Int32 nRecords;
OFF_T *vvrOffset;
{
  CDFstatus pStatus = CDF_OK; struct VVRstruct64 VVR;
  VVR.RecordSize = (OFF_T) (VVR_BASE_SIZE64 + 
		   ((OFF_T)nRecords) * Var->NphyRecBytes);
  VVR.RecordType = VVR_;
  if (!sX(AllocateIR64(CDF,VVR.RecordSize,vvrOffset),&pStatus)) return pStatus;
  if (!sX(WriteVVR64(CDF->fp,*vvrOffset,
		     VVR_RECORDx,&VVR,
		     VVR_NULL),&pStatus)) return pStatus;
  return pStatus;
}

/******************************************************************************
* AllocateCVVR.
******************************************************************************/

static CDFstatus AllocateCVVR (CDF, cSize, xSize, cvvrOffset)
struct CDFstruct *CDF;
OFF_T cSize;
OFF_T xSize;
OFF_T *cvvrOffset;
{
  CDFstatus pStatus = CDF_OK; struct CVVRstruct64 CVVR;
  CVVR.RecordSize = (OFF_T) (CVVR_BASE_SIZE64 + cSize + xSize);
  CVVR.RecordType = CVVR_;
  CVVR.rfuA = (Int32) 0;
  CVVR.cSize = cSize;
  if (!sX(AllocateIR64(CDF,CVVR.RecordSize,cvvrOffset),&pStatus)) return pStatus;
  if (!sX(WriteCVVR64(CDF->fp,*cvvrOffset,
		      CVVR_RECORDx,&CVVR,
		      CVVR_NULL),&pStatus)) return pStatus;
  return pStatus;
}

/******************************************************************************
* UpdateVXRtailInVDR64.
******************************************************************************/

STATICforIDL CDFstatus UpdateVXRtailInVDR64 (CDF, Var)
struct CDFstruct *CDF;
struct VarStruct *Var;
{
  CDFstatus pStatus = CDF_OK; struct VXRstruct64 VXR;
  OFF_T vxrOffset; Int32 irType; int lastEntryN;
  /****************************************************************************
  * Read the VXR head.  If zero, set the VXR tail to zero.
  ****************************************************************************/
  if (!sX(ReadVDR64(CDF,CDF->fp,Var->VDRoffset64,Var->zVar,
		    VDR_VXRHEAD,&vxrOffset,
		    VDR_NULL),&pStatus)) return pStatus;
  if (vxrOffset == (OFF_T) ZERO_OFFSET64) {
    if (!sX(WriteVDR64(CDF,CDF->fp,Var->VDRoffset64,Var->zVar,
		       VDR_VXRTAIL,&vxrOffset,
		       VDR_NULL),&pStatus)) return pStatus;
    return pStatus;
  }
  /****************************************************************************
  * Otherwise, read to the VXR tail...
  ****************************************************************************/
  for (;;) {
     if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		       VXR_RECORD,&VXR,
		       VXR_NULL),&pStatus)) return pStatus;
     while (VXR.VXRnext != (OFF_T) ZERO_OFFSET64) {
       vxrOffset = VXR.VXRnext;
       if (!sX(ReadVXR64(CDF->fp,vxrOffset,
		         VXR_RECORD,&VXR,
		         VXR_NULL),&pStatus)) return pStatus;
     }
     lastEntryN = (int) (VXR.NusedEntries - 1);
     if (!sX(ReadIrType64(CDF->fp,
			  VXR.Offset[lastEntryN],
			  &irType),&pStatus)) return pStatus;
     switch (irType) {
       case VXR_:
	 vxrOffset = VXR.Offset[lastEntryN];
	 break;
       case VVR_:
       case CVVR_:
	 if (!sX(WriteVDR64(CDF,CDF->fp,Var->VDRoffset64,Var->zVar,
			    VDR_VXRTAIL,&vxrOffset,
			    VDR_NULL),&pStatus)) return pStatus;
	 return pStatus;
       default:
	 return CORRUPTED_V3_CDF;
     }
  }
}



syntax highlighted by Code2HTML, v. 0.9.1