/* vim: set expandtab ts=4 sw=4: */
/*
* You may redistribute this program and/or modify it under the terms of
* the GNU General Public License as published by the Free Software Foundation,
* either version 3 of the License, or (at your option) any later version.
*
* 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.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#ifndef LabelSplicer_H
#define LabelSplicer_H
#include "switch/NumberCompress.h"
#include "util/Bits.h"
#include
#include
/**
* Convert a full path (represention we can use) to a representaiton which a node
* along that path can use.
*
* @param fullPath the full route to the destination
* @param midPath a path to a node which falls somewhere within fullPath
* @return a version of fullPath which is sutible for use by node at midPath
*/
static inline uint64_t LabelSplicer_unsplice(uint64_t fullPath, uint64_t midPath)
{
return fullPath >> Bits_log2x64(midPath);
}
/**
* Splice a label and a label fragment together.
*
*/
static inline uint64_t LabelSplicer_splice(uint64_t goHere, uint64_t viaHere)
{
uint64_t log2ViaHere = Bits_log2x64(viaHere);
if (Bits_log2x64(goHere) + log2ViaHere > 60) {
// Too big, can't splice.
return UINT64_MAX;
}
return ((goHere ^ 1) << log2ViaHere) ^ viaHere;
}
/**
* Get the label for a particular destination from a given source.
* This needs to be called before handing out a label because if a source interface is
* represented using more bits than the destination interface, the destination interface
* must be padded out so that the switch will find the source and destination labels compatable.
*
* @param target the label for the location to send to in host byte order.
* @param whoIsAsking the label for the node which we are sending the target to in host byte order.
* @return the modified target for that node in host byte order.
*/
static inline uint64_t LabelSplicer_getLabelFor(uint64_t target, uint64_t whoIsAsking)
{
uint32_t targetBits = NumberCompress_bitsUsedForLabel(target);
uint32_t whoIsAskingBits = NumberCompress_bitsUsedForLabel(whoIsAsking);
if (targetBits >= whoIsAskingBits) {
return target;
}
uint32_t targetIfaceNum = NumberCompress_getDecompressed(target, targetBits);
return ((target & (UINT64_MAX << targetBits)) << (whoIsAskingBits - targetBits))
| NumberCompress_getCompressed(targetIfaceNum, whoIsAskingBits);
}
/**
* Determine if the node at the end of the given label is one hop away.
*
* @param label the label to test in host byte order.
* @return true if the node is 1 hop away, false otherwise.
*/
static inline bool LabelSplicer_isOneHop(uint64_t label)
{
return (int)NumberCompress_bitsUsedForLabel(label) == Bits_log2x64(label);
}
/**
* Determine if the route to one node passes through another node.
* Given:
* 1. alice->bob->charlie->fred->bob
* 2. alice->bob
* 1 routes through 2.
*
* @param destination the node to route to.
* @param midPath the node which might be in the middle of the route.
* @return true if midPath is in the middle of the route to destination.
*/
static inline bool LabelSplicer_routesThrough(uint64_t destination, uint64_t midPath)
{
if (midPath > destination) {
return false;
} if (midPath < 2) {
return true;
}
uint64_t mask = UINT64_MAX >> (64 - Bits_log2x64(midPath));
return (destination & mask) == (midPath & mask);
}
#endif