-- file Pass3V.Mesa
-- last modified by Satterthwaite, March 11, 1983 12:33 pm

DIRECTORY
  A3: TYPE USING [
    CanonicalType, LongPath, MarkedType, OperandType, TargetType, TypeForTree],
  Alloc: TYPE USING [Notifier],
  ComData: TYPE USING [ownSymbols, seAnon, textIndex, typeAtomRecord, typeBOOL],
  Copier: TYPE USING [CopyUnion],
  Log: TYPE USING [Error, ErrorHti, ErrorSei, ErrorTree, Warning],
  P3: TYPE USING [
    Attr, NarrowOp, NPUse, phraseNP,
    BaseTree, EnterType, Exp, FindSe, FirstId, ForceType, OpenPointer,
    PopCtx, PushCtx, PushHtCtx, PushRecordCtx, RAttr, Rhs, RPop, RPush, RType,
    Scope, SealRefStack, SearchCtxList, TopCtx, UnsealRefStack, UpdateTreeAttr, VoidExp],
  P3S: TYPE USING [ImplicitInfo, implicit, implicitRecord, safety],
  Symbols: TYPE USING [
    Base, Name, Type, ISEIndex, CSEIndex, RecordSEIndex, CTXIndex,
    nullName, nullType, ISENull, CTXNull, typeANY, typeTYPE, seType, ctxType],
  SymbolOps: TYPE USING [
    NextSe, NormalType, RCType, ReferentType, TypeForm, TypeLink, TypeRoot, UnderType],
  Tree: TYPE USING [Base, Index, Link, Map, Null, Scan, treeType],
  TreeOps: TYPE USING [
    GetHash, GetNode, ListHead, ListTail, OpName, PopTree, PushTree, PushNode, PushSe,
    ScanList, SetAttr, SetInfo, UpdateList],
  Types: TYPE USING [Equivalent];

Pass3V: PROGRAM
    IMPORTS
      A3, Copier, Log, P3, P3S, SymbolOps, TreeOps, Types,
      dataPtr: ComData
    EXPORTS P3 = { 
  OPEN SymbolOps, A3, P3, Symbols, TreeOps;

-- tables defining the current symbol table

  tb: Tree.Base;		-- tree base
  seb: Symbols.Base;		-- se table
  ctxb: Symbols.Base;		-- context table

  VRNotify: PUBLIC Alloc.Notifier = {
    -- called whenever the main symbol table is repacked
    tb ← base[Tree.treeType];  seb ← base[seType];  ctxb ← base[ctxType]};


-- finding union and discriminated types
-- N. B. the following two entries cannot assume well-formed type links

  VariantUnionType: PUBLIC PROC [type: Type] RETURNS [CSEIndex] = {
    rType: CSEIndex = ConsType[type];
    RETURN [WITH seb[rType] SELECT FROM
      record =>
	IF hints.variant
	  THEN ConsType[TypeForSe[UnionField[LOOPHOLE[rType, RecordSEIndex]]]]
	  ELSE typeANY,
      ENDCASE => typeANY]};


  SelectVariantType: PUBLIC PROC [type: Type, tag: Name] RETURNS [sei: ISEIndex] = {
    matched: BOOL;
    vType: CSEIndex = VariantUnionType[type];
    WITH seb[vType] SELECT FROM
      union => [matched, sei] ← SearchCtxList[tag, caseCtx];
      ENDCASE => matched ← FALSE;
    IF ~matched THEN sei ← ISENull;
    RETURN};


  SequenceField: PUBLIC PROC [rSei: RecordSEIndex] RETURNS [ISEIndex] = {
    sei: ISEIndex = UnionField[rSei];
    RETURN [IF TypeForm[seb[sei].idType] = sequence THEN sei ELSE ISENull]};


 -- auxiliary procedures (for avoiding UnderType when potentially unsafe)

  UnionField: PROC [rSei: RecordSEIndex] RETURNS [ISEIndex] = {
    sei, root, next: ISEIndex;
    ctx: CTXIndex = seb[rSei].fieldCtx;
    IF ctxb[ctx].ctxType = simple THEN
      FOR sei ← ctxb[ctx].seList, next UNTIL sei = ISENull DO
	next ← NextSe[sei];
	IF next = ISENull THEN RETURN [sei];
	ENDLOOP
    ELSE {	-- defined in another module, UnderType is safe
      repeated: BOOL ← FALSE;
      DO
	sei ← root ← ctxb[ctx].seList;
	DO
	  IF sei = ISENull THEN EXIT;
	  SELECT TypeForm[seb[sei].idType] FROM
	    union, sequence => RETURN [sei];
	    ENDCASE;
	  IF (sei ← NextSe[sei]) = root THEN EXIT;
	  ENDLOOP;
	IF repeated THEN EXIT;
	Copier.CopyUnion[seb[rSei].fieldCtx];  repeated ← TRUE;
	ENDLOOP};
    RETURN [dataPtr.seAnon]};

  ResolveId: PROC [name: Name, ctx: CTXIndex] RETURNS [sei: ISEIndex] = {
    currentCtx: CTXIndex = TopCtx[];
    IF ctx = currentCtx THEN sei ← FindSe[name].symbol
    ELSE {PopCtx[]; sei ← ResolveId[name, ctx]; PushCtx[currentCtx]};
    RETURN};

  TypeForSe: PROC [sei: ISEIndex] RETURNS [type: Type] = INLINE {
    node: Tree.Index;
    t: Tree.Link;
    IF seb[sei].mark3 THEN RETURN [seb[sei].idType];
    node ← seb[sei].idValue;
    IF tb[node].name # decl THEN RETURN [typeTYPE];
    t ← tb[node].son[2];
    type ← WITH t SELECT FROM
      hash => ResolveId[index, seb[sei].idCtx],
      symbol => index,
      subtree => tb[index].info,
      ENDCASE => typeANY;
    RETURN};

  ConsType: PROC [type: Type] RETURNS [CSEIndex] = {
    WITH se: seb[type] SELECT FROM
      id =>
	IF se.mark3 THEN
	  RETURN [IF se.idType # typeTYPE THEN typeANY ELSE ConsType[se.idInfo]]
	ELSE {
	  node: Tree.Index = se.idValue;
	  RETURN [IF tb[node].name # typedecl
	    THEN typeANY
	    ELSE ConsType[ResolveTreeType[tb[node].son[2], se.idCtx]]]};
      cons => RETURN [LOOPHOLE[type, CSEIndex]];
      ENDCASE => ERROR};

  ResolveTreeType: PROC [t: Tree.Link, ctx: CTXIndex] RETURNS [type: Type] = {
    WITH t SELECT FROM
      hash => type ← ResolveId[index, ctx];
      symbol => type ← index;
      subtree => {
	node: Tree.Index = index;
	type ← IF tb[node].info # nullType
	  THEN tb[node].info
	  ELSE
	    SELECT tb[node].name FROM
	      discrimTC =>
		WITH tb[node].son[2] SELECT FROM
		  hash => SelectVariantType[ResolveTreeType[tb[node].son[1], ctx], index],
		  ENDCASE => ERROR,
	      ENDCASE => ERROR};
      ENDCASE => ERROR;
    RETURN};


 -- type discrimination

  DiscriminatedType: PUBLIC PROC [baseType: CSEIndex, t: Tree.Link]
      RETURNS [type: CSEIndex] = {
    IF t = Tree.Null THEN type ← P3S.implicitRecord
    ELSE
      WITH t SELECT FROM
	subtree => {
	  node: Tree.Index = index;
	  temp: Tree.Link;
	  SELECT tb[node].name FROM
	    union => {
	      type ← WITH tb[node].son[1] SELECT FROM symbol => UnderType[index], ENDCASE => ERROR;
	      WITH seb[type] SELECT FROM
		record =>
		  IF hints.variant AND tb[node].son[2] # Tree.Null
		   AND (temp←ListTail[tb[node].son[2]]) # Tree.Null THEN
		    type ← DiscriminatedType[type, temp];
		ENDCASE => ERROR};
	    dollar => type ← OperandType[tb[node].son[1]];
	    dot => {
	      subType: CSEIndex = NormalType[OperandType[tb[node].son[1]]];
	      type ← WITH seb[subType] SELECT FROM ref => UnderType[refType], ENDCASE => ERROR};
	    assignx => type ← DiscriminatedType[baseType, tb[node].son[2]];
	    ENDCASE => type ← baseType};
	ENDCASE => type ← baseType;
    RETURN};


 -- discrimination operations
 
  Narrowing: PUBLIC PROC [type, target: CSEIndex] RETURNS [op: NarrowOp←[]] = {
    typeL: CSEIndex ← target;
    typeR: CSEIndex ← type;
    nextL, nextR: Type;
    IF ~Types.Equivalent[[dataPtr.ownSymbols, type], [dataPtr.ownSymbols, target]] THEN
     DO
      WITH tR: seb[typeR] SELECT FROM
	any => {
	  IF ~op.indirect THEN op.error ← TRUE;
	  WITH tL: seb[typeL] SELECT FROM
	    any => EXIT;
	    opaque => {
	      op.rtTest ← TRUE;
	      IF typeL # dataPtr.typeAtomRecord THEN op.unImpl ← TRUE;
	      EXIT};
	    ENDCASE => {
	      op.rtTest ← TRUE;
	      IF ~Discriminated[typeL] THEN EXIT;
	      nextL ← typeL;  nextR ← TypeRoot[typeL]}};
	record =>
	  WITH tL: seb[typeL] SELECT FROM
	    record => {
	      IF Types.Equivalent[[dataPtr.ownSymbols, typeL], [dataPtr.ownSymbols, typeR]]
	        THEN EXIT;
	      WITH vL: tL SELECT FROM
	        linked => {
		  uType: CSEIndex = VariantUnionType[vL.linkType];
		  WITH u: seb[uType] SELECT FROM
		    union => IF u.controlled THEN op.tagTest ← TRUE ELSE op.computed ← TRUE;
		    ENDCASE => op.error ← TRUE;
		  nextL ← vL.linkType;  nextR ← typeR};
		ENDCASE => op.error ← TRUE};
	    ENDCASE => op.error ← TRUE;
	ref =>
	  WITH tL: seb[typeL] SELECT FROM
	    ref => {
	      IF op.indirect OR (tL.counted # tR.counted) OR (tR.readOnly AND ~tL.readOnly)
	        THEN op.error ← TRUE;
	      op.indirect ← TRUE;
	      nextL ← tL.refType;  nextR ← tR.refType};
	    ENDCASE => op.error ← TRUE;
	transfer =>
	  WITH tL: seb[typeL] SELECT FROM
	    transfer => {
	      IF op.indirect OR tL.mode # tR.mode OR tL.safe # tR.safe THEN
	        op.error ← TRUE;
	      SELECT tL.mode FROM
		proc, signal, error => NULL;
		ENDCASE => op.error ← TRUE;
	      IF TypeForm[tL.typeIn] = any OR TypeForm[tL.typeOut] = any THEN
	        op.error ← TRUE;	-- for now
	      IF TypeForm[tR.typeIn] = any THEN
	        op.rtTest ← TRUE
	      ELSE IF ~Types.Equivalent[
	          [dataPtr.ownSymbols, tL.typeIn],
		  [dataPtr.ownSymbols, tR.typeIn]] THEN op.error ← TRUE;
	      IF TypeForm[tR.typeOut] = any THEN
	        op.rtTest ← TRUE
	      ELSE IF ~Types.Equivalent[
	          [dataPtr.ownSymbols, tL.typeOut],
		  [dataPtr.ownSymbols, tR.typeOut]] THEN op.error ← TRUE;
	      EXIT};
	    ENDCASE => op.error ← TRUE;
	long =>
	  WITH tL: seb[typeL] SELECT FROM
	    long => {nextL ← tL.rangeType; nextR ← tR.rangeType};
	    ENDCASE => op.error ← TRUE;
	ENDCASE => {
	  IF Types.Equivalent[[dataPtr.ownSymbols, typeL], [dataPtr.ownSymbols, typeR]]
	    THEN EXIT;
	  op.error ← TRUE};
      IF op.error THEN EXIT;
      typeL ← UnderType[nextL];  typeR ← UnderType[nextR];
      ENDLOOP;
    RETURN};
    
  Discriminated: PROC [type: CSEIndex] RETURNS [BOOL] = INLINE {
    RETURN [TypeLink[type] # nullType]};  -- check that at tag exists?
    
 -- binding of variant records

  Discrimination: PUBLIC PROC [node: Tree.Index, selection: Tree.Map] = {
    OPEN tb[node];
    copy: BOOL = (OpName[ListHead[son[3]]] = ditem);
    type: Type;
    nType, subType: CSEIndex;
    vCtx: CTXIndex;
    base, discBase: Tree.Link;
    attr: Attr;
    entryNP: NPUse ← none;
    unreachable: BOOL ← FALSE;

    BindError: PROC = {
      IF son[2] # Tree.Null THEN son[2] ← VoidExp[son[2]]; vCtx ← CTXNull};

    PushCommonCtx: PROC = {
      SELECT TRUE FROM
	copy OR (seb[nType].typeTag # record) => PushCtx[CTXNull];
	(baseId = nullName) => PushRecordCtx[LOOPHOLE[nType], base, indirect];
	ENDCASE => PushHtCtx[baseId, base, indirect]};

    BindItem: Tree.Scan = {
      subNode: Tree.Index = GetNode[t];
      saveIndex: CARDINAL = dataPtr.textIndex;
      IF tb[subNode].name = ditem THEN {
	declNode: Tree.Index = GetNode[tb[subNode].son[1]];
	declType: CSEIndex;
	Item: Tree.Map = {phraseNP ← entryNP; v ← selection[t]};
	op: NarrowOp;
	dataPtr.textIndex ← tb[declNode].info;
	IF unreachable THEN {Log.Warning[unreachable]; unreachable ← FALSE};
	Scope[subNode, Item];
	declType ← UnderType[TypeForTree[tb[declNode].son[2]]];
	op ← Narrowing[subType, declType];
	SELECT TRUE FROM
	  ~copy => Log.Error[discrimForm];
	  op.error => Log.ErrorSei[typeClash, FirstId[declNode]];
	  op.computed => Log.ErrorTree[missingBinding, base];
	  op.unImpl => Log.Warning[unimplemented];
	  ~(op.rtTest OR op.tagTest) => unreachable ← TRUE;
	  ENDCASE;
	tb[subNode].attr1 ← op.indirect;
	IF (tb[subNode].attr2 ← op.rtTest) THEN EnterType[MarkedType[declType]];
	tb[subNode].attr3 ← op.tagTest}
      ELSE {
	vType: CSEIndex;
	dataPtr.textIndex ← tb[subNode].info;
	IF copy THEN {Log.Error[discrimForm]; attr3 ← FALSE};
	[tb[subNode].son[1], vType] ← BindTest[tb[subNode].son[1], vCtx];
	IF vType = typeANY THEN PushCommonCtx[]
	ELSE {
	  WITH discBase SELECT FROM
	    subtree => tb[index].info ← vType;
	    ENDCASE => ERROR;
	  IF baseId = nullName THEN PushRecordCtx[LOOPHOLE[vType], discBase, FALSE]
	  ELSE PushHtCtx[baseId, discBase, FALSE]};
	phraseNP ← entryNP;
	tb[subNode].son[2] ← selection[tb[subNode].son[2]];
	PopCtx[];
	tb[subNode].attr1 ← TRUE};
      dataPtr.textIndex ← saveIndex};

    saveImplicit: P3S.ImplicitInfo = P3S.implicit;
    idNode: Tree.Index = GetNode[son[1]];
    baseId: Name = GetHash[tb[idNode].son[1]];
    indirect: BOOL;
    SealRefStack[];
    base ← tb[idNode].son[2] ← Exp[tb[idNode].son[2], typeANY];
    type ← RType[];  attr ← RAttr[];  RPop[];
    UnsealRefStack[];
    subType ← CanonicalType[type];
    IF subType # UnderType[type] THEN
      tb[idNode].son[2] ← ForceType[tb[idNode].son[2], subType];
    nType ← NormalType[subType];
    P3S.implicit ← [tree: base, type: subType, attr: attr];
    IF (attr3 ← copy) THEN {
      P3S.implicit.attr.noAssign ← P3S.implicit.attr.noXfer ← TRUE;
      SELECT TypeForm[nType] FROM
        ref => {
	  attr2 ← (TypeForm[ReferentType[nType]] = any);
	  indirect ← TRUE};
	transfer => {attr2 ← TRUE; indirect ← FALSE};
	ENDCASE => {attr2 ← FALSE; indirect ← FALSE};
      IF baseId # nullName THEN Log.Error[discrimForm]}
    ELSE {
      long: BOOL;
      WITH t: seb[nType] SELECT FROM
	ref => {
	  indirect ← TRUE;
	  [base, type] ← OpenPointer[base, subType];
	  subType ← OperandType[base];  nType ← NormalType[type];
	  long ← seb[subType].typeTag = long};
	ENDCASE => {indirect ← FALSE; long ← LongPath[base]};
      IF P3S.safety = checked AND RCType[nType] # none THEN
        Log.ErrorTree[unsafeSelection, base];
      WITH seb[nType] SELECT FROM
	record => {
	  tb[idNode].son[2] ← base ← BaseTree[base, subType];
	  IF hints.variant THEN {
	    uType: CSEIndex = VariantUnionType[nType];
	    WITH u: seb[uType] SELECT FROM
	      union => {
		tagType: CSEIndex = UnderType[seb[u.tagSei].idType];
		vCtx ← u.caseCtx;
		IF son[2] = Tree.Null THEN {
		  IF ~u.controlled THEN Log.ErrorTree[missingBinding, base];
		  [] ← UpdateTreeAttr[base];  entryNP ← phraseNP;
		  PushTree[base];  PushSe[u.tagSei];
		  PushNode[IF indirect THEN dot ELSE dollar, 2];
		  SetInfo[tagType]; SetAttr[2, long]; son[2] ← PopTree[]}
		ELSE {
		  IF u.controlled THEN Log.ErrorTree[spuriousBinding, son[2]];
		  PushCommonCtx[];
		  son[2] ← Rhs[son[2], TargetType[tagType]];
		  entryNP ← phraseNP;  RPop[];
		  PopCtx[]}};
	      ENDCASE => {Log.Error[noAccess]; BindError[]}}
	  ELSE {Log.ErrorTree[noVariants, tb[idNode].son[2]]; BindError[]};
	  PushTree[base];
	  IF indirect THEN {PushNode[uparrow, 1]; SetAttr[2, long]}
	  ELSE PushNode[cast, 1];
	  discBase ← PopTree[]};
	ENDCASE => {
	  Log.ErrorTree[noVariants, tb[idNode].son[2]];  BindError[];
	  discBase ← Tree.Null}};
    attr1 ← indirect;
    ScanList[son[3], BindItem];
    PushCommonCtx[]; phraseNP ← entryNP; son[4] ← selection[son[4]]; PopCtx[];
    RPush[nullType, attr];
    P3S.implicit ← saveImplicit};


  BindTest: PROC [t: Tree.Link, vCtx: CTXIndex]
      RETURNS [val: Tree.Link, vType: CSEIndex] = {
    mixed: BOOL ← FALSE;

    TestItem: Tree.Map = {
      WITH t SELECT FROM
	subtree => {
	  subNode: Tree.Index = index;
	  SELECT tb[subNode].name FROM
	    relE =>
	      WITH tb[subNode].son[2] SELECT FROM
		hash => {
		  iType: ISEIndex;
		  uType: CSEIndex;
		  found: BOOL;
		  [found, iType] ← SearchCtxList[index, vCtx];
		  IF found THEN {
		    uType ← UnderType[iType];
		    tb[subNode].son[2] ← [symbol[index: iType]];
		    SELECT vType FROM
		      uType => NULL;
		      typeANY => vType ← uType;
		      ENDCASE => mixed ← TRUE}
		  ELSE IF vCtx # CTXNull THEN Log.ErrorHti[unknownVariant, index];
		  tb[subNode].info ← dataPtr.typeBOOL;
		  tb[subNode].attr1 ← tb[subNode].attr2 ← FALSE;
		  v ← t};
		ENDCASE => {
		  v ← Rhs[t, dataPtr.typeBOOL];  RPop[];
		  Log.ErrorTree[nonVariantLabel, t]};
	    ENDCASE => {
	      v ← Rhs[t, dataPtr.typeBOOL];  RPop[];
	      Log.ErrorTree[nonVariantLabel, t]}};
	ENDCASE => ERROR;
      RETURN};

    vType ← typeANY;  val ← UpdateList[t, TestItem];
    IF mixed THEN vType ← typeANY;
    RETURN};

  }.