-- File: SimpleNSIOEthernetImpl.mesa - last edit:
-- AOF                  9-Feb-88 10:10:04
-- HGM                 24-Oct-85 17:54:26, Remove SLOW Dove "bug fix"
-- Copyright (C) 1983, 1985 , 1988 by Xerox Corporation. All rights reserved. 

-- This module implements SimpleNSIO on the Ethernet (NOT the EthernetOne). It supplies level one and level zero encapsulation.

-- NOTE: The module SimpleNSIOEthernetOneImpl is a near-copy of this one. IF YOU MAKE A CHANGE HERE, it probably should be made to it too. SimplePUPIOEthernetOneImpl is another, somewhat less-similar module.

DIRECTORY
  BootChannel USING [Result],
  Checksum USING [ComputeChecksum],
  Device USING [Type],
  DeviceTypes USING [ethernet],
  IEEE8023 USING [EncapObject, EthernetCRC, minWordsPerEthernetPacket],
  Environment USING [Base, bytesPerWord, first64K],
  EthernetFace USING [
    AddCleanup, controlBlockSize, DeviceHandle, GetNextDevice, GetPacketLength,
    GetStatus, globalStateSize, nullDeviceHandle, QueueInput, QueueOutput,
    RemoveCleanup, TurnOff, TurnOn, GlobalStatePtr],
  GermOps USING [GermWorldError],
  Inline USING [LowHalf],
  NSTypes USING [BufferBody, bytesPerIDPHeader, maxIDPWordsPerPacket, PacketType],
  PilotMP USING [cGermERROR],
  ProcessorFace USING [processorID],
  ResidentHeap USING [FreeNode, MakeNode],
  SimpleNSIO USING [
    ByteCount, cantHandleDevice, Cleanups, timedOutBytes, timedOutResult],
  SpecialSystem USING [
    HostNumber, NetworkNumber, nullNetworkNumber, SocketNumber],
  System USING [
    GetClockPulses, Microseconds, MicrosecondsToPulses, NetworkAddress,
    nullHostNumber, nullNetworkNumber, nullSocketNumber, Pulses, SocketNumber];

SimpleNSIOEthernetImpl: PROGRAM
  IMPORTS
    Checksum, EthernetFace, GermOps, Inline, ProcessorFace, ResidentHeap, System
  EXPORTS SimpleNSIO, System
  SHARES GermOps =
  BEGIN
  
  -- PARAMETERS:
  
  sendTimeOut: System.Microseconds = 3000000;

  -- TYPES AND VARIABLES:

  HostNumber: PUBLIC --System-- TYPE = SpecialSystem.HostNumber;
  NetworkNumber: PUBLIC --System-- TYPE = SpecialSystem.NetworkNumber;
  SocketNumber: PUBLIC --System-- TYPE = SpecialSystem.SocketNumber;

  NetworkAddress: TYPE = System.NetworkAddress;

  ByteCount: TYPE = SimpleNSIO.ByteCount;
  
  -- Level zero Ethernet packet format:
  LevelZeroPacket: TYPE = MACHINE DEPENDENT RECORD [
    encapsulation(0): ethernet IEEE8023.EncapObject,
    levelOnePacket(7): NSTypes.BufferBody];
  bufferAlign: CARDINAL = 3;  -- align encapsulation to this boundary..
  bufferMod: CARDINAL = 4;  -- .. modulo this.

  slop: CARDINAL = 12;  -- Why this? see Footnote (1) beneath LOG.
  bufferSize: CARDINAL =
    SIZE[ethernet IEEE8023.EncapObject] +
    NSTypes.maxIDPWordsPerPacket +
    IEEE8023.EthernetCRC.SIZE +
    slop;
  buffer: LONG POINTER TO LevelZeroPacket;

  longIocb: LONG POINTER ← NIL;
  globalStorage: EthernetFace.GlobalStatePtr;
  device: EthernetFace.DeviceHandle ← EthernetFace.nullDeviceHandle;

  returnData: RECORD [
    returnTo, returnedFrom: NetworkAddress, returnVia: HostNumber];


  --==============================
  -- Public Procedures
  --==============================

  GetRawBufferSize: PUBLIC PROCEDURE RETURNS [rawBufferSize: CARDINAL] = {
    RETURN[bufferSize + bufferMod-1]};

  Initialize: PUBLIC PROC [
    deviceType: Device.Type, deviceOrdinal: CARDINAL, rawBuffer: LONG POINTER,
    cleanups: SimpleNSIO.Cleanups]
    RETURNS [
      result: BootChannel.Result, bufferBody: LONG POINTER TO NSTypes.BufferBody] =
    BEGIN
    IF device # EthernetFace.nullDeviceHandle THEN Error[];  -- already initialized
    IF deviceType # DeviceTypes.ethernet THEN
      RETURN[SimpleNSIO.cantHandleDevice, NIL];
    THROUGH [0..deviceOrdinal] DO
      IF (device ← EthernetFace.GetNextDevice[device]) =
        EthernetFace.nullDeviceHandle THEN
	RETURN[SimpleNSIO.cantHandleDevice, NIL];
      ENDLOOP;
    longIocb ← @Environment.first64K[
      ResidentHeap.MakeNode[EthernetFace.controlBlockSize, a4].node];
    buffer ← LOOPHOLE[((LOOPHOLE[rawBuffer, LONG CARDINAL] + bufferMod-1)/bufferMod)*bufferMod
      + bufferAlign];
    IF EthernetFace.globalStateSize # 0 THEN
      [globalStorage, ] ← ResidentHeap.MakeNode[EthernetFace.globalStateSize, a4];
    IF cleanups = doCleanup THEN EthernetFace.AddCleanup[device];
    EtherTurnOn[];
    RETURN[[ok[]], @buffer.levelOnePacket];
    END;  --Initialize--

  Finalize: PUBLIC PROC [cleanups: SimpleNSIO.Cleanups] =
    BEGIN
    IF device = EthernetFace.nullDeviceHandle THEN Error[];  -- not initialized
    EthernetFace.TurnOff[device];
    IF cleanups = doCleanup THEN EthernetFace.RemoveCleanup[device];
    device ← EthernetFace.nullDeviceHandle;
    [] ← ResidentHeap.FreeNode[Inline.LowHalf[longIocb]];
    longIocb ← NIL;
    IF EthernetFace.globalStateSize # 0 THEN
      [] ← ResidentHeap.FreeNode[globalStorage];
    END;

  ReceivePacket: PUBLIC PROC [
    getFrom: NetworkAddress, mySocket: System.SocketNumber,
    timeout: System.Microseconds]
    RETURNS [
      result: BootChannel.Result, dataBytes: ByteCount, type: NSTypes.PacketType,
      source, destination: LONG POINTER TO NetworkAddress] =
    BEGIN
    startTime: System.Pulses = System.GetClockPulses[];
    IF device = EthernetFace.nullDeviceHandle THEN Error[];  -- not initialized
    DO  --UNTIL packet arrives on requested socket--
      EthernetFace.QueueInput[
        device, @buffer.encapsulation, bufferSize, longIocb];
      UNTIL EthernetFace.GetStatus[longIocb] # pending DO
        IF TimeExpired[timeout: timeout, startTime: startTime] THEN
          BEGIN
          EthernetFace.TurnOff[device];  -- abort the input.
          EtherTurnOn[];
          RETURN[
	    dataBytes: SimpleNSIO.timedOutBytes,
	    type: TRASH, source: TRASH, destination: TRASH,
	    result: SimpleNSIO.timedOutResult];
          END;
        ENDLOOP;
      IF EthernetFace.GetStatus[longIocb] ~= ok THEN LOOP;
      IF buffer.encapsulation.ethernetType # ns THEN LOOP;
      IF ~TestChecksum[@buffer.levelOnePacket] THEN LOOP;
      IF buffer.levelOnePacket.destination.host ~= ProcessorFace.processorID THEN LOOP;
      IF getFrom.net ~= System.nullNetworkNumber
        AND getFrom.net ~= buffer.levelOnePacket.source.net THEN LOOP;
      IF getFrom.host ~= System.nullHostNumber
        AND getFrom.host ~= buffer.levelOnePacket.source.host THEN LOOP;
      IF getFrom.socket ~= System.nullSocketNumber
        AND getFrom.socket ~= buffer.levelOnePacket.source.socket THEN LOOP;
      IF mySocket ~= System.nullSocketNumber
        AND mySocket ~= buffer.levelOnePacket.destination.socket THEN LOOP;
      IF EthernetFace.GetPacketLength[longIocb] <
        LevelZeroWordsFromLevelOneBytes[
          buffer.levelOnePacket.pktLength] THEN LOOP;
      IF buffer.levelOnePacket.pktLength >= NSTypes.bytesPerIDPHeader THEN EXIT;
      ENDLOOP;
    returnData.returnTo ← buffer.levelOnePacket.source;
    returnData.returnedFrom ← buffer.levelOnePacket.destination;
    returnData.returnVia ← buffer.encapsulation.ethernetSource;
    RETURN[
      dataBytes: buffer.levelOnePacket.pktLength - NSTypes.bytesPerIDPHeader,
      type: buffer.levelOnePacket.packetType,
      source: @buffer.levelOnePacket.source,
      destination: @buffer.levelOnePacket.destination,
      result: [ok[]]];
    END;  --ReceivePacket--

  ReturnPacket: PUBLIC PROC [dataBytes: ByteCount, type: NSTypes.PacketType]
    RETURNS [result: BootChannel.Result] =
    BEGIN
    IF device = EthernetFace.nullDeviceHandle THEN Error[];  -- not initialized
    buffer.encapsulation.ethernetDest ← returnData.returnVia;
    buffer.levelOnePacket.source ← returnData.returnedFrom;
    buffer.levelOnePacket.destination ← returnData.returnTo;
    RETURN SendBuffer[dataBytes, type];
    END;

  SendPacket: PUBLIC PROC [
    dataBytes: ByteCount, type: NSTypes.PacketType, sourceSocket: SocketNumber,
    dest: NetworkAddress]
    RETURNS [result: BootChannel.Result] =
    BEGIN
    IF device = EthernetFace.nullDeviceHandle THEN Error[];  -- not initialized;
    buffer.encapsulation.ethernetDest ← dest.host;
    buffer.levelOnePacket.source ← [
      net: SpecialSystem.nullNetworkNumber, host: ProcessorFace.processorID,
      socket: sourceSocket];
    buffer.levelOnePacket.destination ← dest;
    RETURN SendBuffer[dataBytes, type];
    END;

  --==============================
  -- Private Procedures
  --==============================

  Error: PROC = {GermOps.GermWorldError[PilotMP.cGermERROR]};

  SendBuffer: PROC [dataBytes: ByteCount, type: NSTypes.PacketType]
    RETURNS [result: BootChannel.Result] =
    -- Sends buffer; doesn't return till it is sent (but times out if can't send).
    BEGIN
    startTime: System.Pulses;
    buffer.levelOnePacket.pktLength ← dataBytes + NSTypes.bytesPerIDPHeader;
    buffer.levelOnePacket.transportControl ← [
      trace: FALSE, filler: 0, hopCount: 0];
    buffer.levelOnePacket.packetType ← type;
    buffer.encapsulation.ethernetSource ← [ProcessorFace.processorID];
    buffer.encapsulation.ethernetType ← ns;
    SetChecksum[@buffer.levelOnePacket];
    EthernetFace.QueueOutput[
      device, @buffer.encapsulation,
      LevelZeroWordsFromLevelOneBytes[buffer.levelOnePacket.pktLength],
      longIocb];
IF FALSE THEN
    THROUGH [0..1000) DO ENDLOOP;  --we haven't figured out why this helps
    startTime ← System.GetClockPulses[];
    DO  --until sent or timed out
      IF EthernetFace.GetStatus[longIocb] # pending THEN EXIT;
      IF TimeExpired[timeout: sendTimeOut, startTime: startTime] THEN {
        EthernetFace.TurnOff[device]; EtherTurnOn[]; EXIT};
      ENDLOOP;
    RETURN[[ok[]]];
    END;  --SendBuffer--

  SetChecksum: PROC [b: LONG POINTER TO NSTypes.BufferBody] = {
    b.checksum ← Checksum.ComputeChecksum[
      0, (b.pktLength + 1 - 2)/2, @b.pktLength]};

  TestChecksum: PROC [b: LONG POINTER TO NSTypes.BufferBody] RETURNS [BOOLEAN] = {
    RETURN[
      b.checksum = Checksum.ComputeChecksum[
        0, (b.pktLength + 1 - 2)/2, @b.pktLength] OR b.checksum = 177777B]};

  TimeExpired: PROC [timeout, startTime: System.Microseconds] RETURNS [BOOLEAN] =
    BEGIN
    IF timeout = LAST[System.Microseconds] THEN RETURN[FALSE]
    ELSE
      RETURN[
        (System.GetClockPulses[] - startTime) > System.MicrosecondsToPulses[
          timeout]];
    END;

  EtherTurnOn: PROC = {EthernetFace.TurnOn[device, [ProcessorFace.processorID], 0, 0, globalStorage]};

  LevelZeroWordsFromLevelOneBytes: PROC [bodyBytes: CARDINAL]
    RETURNS [wds: CARDINAL] =
    BEGIN
    RETURN[MAX[
      IEEE8023.minWordsPerEthernetPacket, 
      SIZE[ethernet IEEE8023.EncapObject] +
        (bodyBytes + Environment.bytesPerWord - 1)/Environment.bytesPerWord]];
    END;


  -- MAIN body:
  --IF EthernetFace.globalStateSize # 0 THEN Error[];

  END.
  
  
LOG

23-Oct-81 17:04:29   FXH       Created file.
24-Nov-81 11:10:38   FXH       Merged in use of ResidentHeap.
10-Dec-81 11:43:21   AWL      
   MicrosecondsToPulses now in System.  Fixes for network address related things
   now being defined in System.
10-Dec-81 17:11:11   AWL         Export types to System from SpecialSystem.
20-Apr-83 13:10:33   WDK     
   OISCP => NS. Renamed module MiniOISCPDriverImpl.mesa => SimpleNSIOEthernetImpl. Now returns all nonfatal errors to caller. Make compatible with new SimpleNSIO. Now will talk to specified ethernet. Get constants from DriverTypes. Buffer passed in by client.
20-Jul-83  9:58:49   AWL      
   NSTypes.bytesPerPktHeader => NSTypes.bytesPerSppHeader,
   NSTypes.maxBytesPerPkt => NSTypes.maxDataBytesPerSpp.
 9-Aug-83 17:05:49   WDK     
   Previous change was wrong. Should be IDP, not SPP. Use new DriverTypes.EthernetCRC.
 8-Aug-85  9:36:25   AOF
   Process globalStorage when Faces says size is # 0 (Daybreak).
16-Oct-85 20:44:44   AOF
   Add a small delay loop after queue output to the ethernet face.  We don't know why this works, but it covers problems in both the DLion and Daybreak machines.     
   
  
  FOOTNOTE
  added at time of Copyriting  8-Oct-84 18:18:35:
  (1) "-- Hal says we need this. We should ask why." where the global constant ~slop~ is declared.