Rev 2801 |
Blame |
Compare with Previous |
Last modification |
View Log
| RSS feed
/*
* Copyright (C) 2008
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* Create date: 07-Jul-2008
*/
package uk.me.parabola.imgfmt.app.net;
import uk.me.parabola.imgfmt.app.ImgFileWriter;
/**
* A restriction in the routing graph.
*
* There may eventually be several types of these at which point
* we might consider splitting them into several classes. For the
* moment, just simple from-to-via restrictions.
*
* A from-to-via restriction says you can't go along arc "to"
* if you came to node to.getSource() == from.getSource()
* via the inverse arc of "from". We're using the inverse of
* "from" since that has the information we need for writing
* the Table C entry.
*
* @author Robert Vollmert
*/
public class RouteRestriction {
//private static final Logger log = Logger.getLogger(RouteRestriction.class);
// size in bytes
private static final int SIZE = 11;
// first three bytes of the header -- might specify the type of restriction
// and when it is active
private static final int HEADER = 0x004005;
// To specify that a node is given by a relative offset instead
// of an entry to Table B.
private static final int F_INTERNAL = 0x8000;
// the arcs
private final RouteArc from;
private final RouteArc to;
// offset in Table C
private byte offsetSize;
private int offsetC;
// last restriction in a node
private boolean last;
// mask that specifies which vehicle types the restriction doesn't apply to
private final byte exceptMask;
private final boolean exceptPedestrian;
private final boolean exceptEmergency;
public final static byte EXCEPT_CAR = 0x01;
public final static byte EXCEPT_BUS = 0x02;
public final static byte EXCEPT_TAXI = 0x04;
public final static byte EXCEPT_DELIVERY = 0x10;
public final static byte EXCEPT_BICYCLE = 0x20;
public final static byte EXCEPT_TRUCK = 0x40;
// additional flags that can be passed via exceptMask
public final static byte EXCEPT_FOOT = 0x08; // not written as such
public final static byte EXCEPT_EMERGENCY = (byte)0x80; // not written as such
private final static byte SPECIAL_EXCEPTION_MASK = ~(EXCEPT_FOOT|EXCEPT_EMERGENCY);
/**
* Create a route restriction.
*
* @param from The inverse arc of "from" arc.
* @param to The "to" arc.
*/
public RouteRestriction(RouteArc from, RouteArc to, byte exceptMask) {
assert from.getSource().equals(to.getSource()) : "arcs in restriction don't meet";
this.from = from;
this.to = to;
this.exceptMask = (byte)(exceptMask & SPECIAL_EXCEPTION_MASK);
this.exceptPedestrian = (exceptMask & EXCEPT_FOOT) != 0;
this.exceptEmergency = (exceptMask & EXCEPT_EMERGENCY) != 0;
}
private int calcOffset(RouteNode node, int tableOffset) {
int offset = tableOffset - node.getOffsetNod1();
assert offset >= 0 : "node behind start of tables";
assert offset < 0x8000 : "node offset too large";
return offset | F_INTERNAL;
}
/**
* Writes a Table C entry.
*
* @param writer The writer.
* @param tableOffset The offset in NOD 1 of the tables area.
*/
public void write(ImgFileWriter writer, int tableOffset) {
int header = HEADER;
if (exceptPedestrian)
header |= 0x0200;
if (exceptEmergency)
header |= 0x0400;
if(exceptMask != 0)
header |= 0x0800;
writer.put3(header);
if(exceptMask != 0)
writer.put(exceptMask);
int[] offsets = new int[3];
if (from.isInternal())
offsets[0] = calcOffset(from.getDest(), tableOffset);
else
offsets[0] = from.getIndexB();
offsets[1] = calcOffset(to.getSource(), tableOffset);
if (to.isInternal())
offsets[2] = calcOffset(to.getDest(), tableOffset);
else
offsets[2] = to.getIndexB();
for (int offset : offsets)
writer.putChar((char) offset);
writer.put(from.getIndexA());
writer.put(to.getIndexA());
}
/**
* Write this restriction's offset within Table C into a node record.
*/
public void writeOffset(ImgFileWriter writer) {
assert 0 < offsetSize && offsetSize <= 2 : "illegal offset size";
int offset = offsetC;
if (offsetSize == 1) {
assert offset < 0x80;
if (last)
offset |= 0x80;
writer.put((byte) offset);
} else {
assert offset < 0x8000;
if (last)
offset |= 0x8000;
writer.putChar((char) offset);
}
}
/**
* Size in bytes of the Table C entry.
*/
public int getSize() {
int size = SIZE;
if(exceptMask != 0)
++size;
return size;
}
public void setOffsetC(int offsetC) {
this.offsetC = offsetC;
}
public int getOffsetC() {
return offsetC;
}
public void setOffsetSize(byte size) {
offsetSize = size;
}
public void setLast() {
last = true;
}
}