-
Marcel Loose authored
Bug 1310: Moved directory RSPDriver to RSP_Driver and TBBDriver to TBB_Driver in order to sync directory and package names
Marcel Loose authoredBug 1310: Moved directory RSPDriver to RSP_Driver and TBBDriver to TBB_Driver in order to sync directory and package names
Code owners
Assign users and groups as approvers for specific file changes. Learn more.
BWWrite.cc 7.79 KiB
//# BWWrite.cc: implementation of the BWWrite class
//#
//# Copyright (C) 2002-2004
//# ASTRON (Netherlands Foundation for Research in Astronomy)
//# P.O.Box 2, 7990 AA Dwingeloo, The Netherlands, seg@astron.nl
//#
//# This program is free software; you can redistribute it and/or modify
//# it under the terms of the GNU General Public License as published by
//# the Free Software Foundation; either version 2 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, write to the Free Software
//# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
//#
//# $Id$
#include <lofar_config.h>
#include <Common/LofarLogger.h>
#include <Common/hexdump.h>
#include <APL/RSP_Protocol/EPA_Protocol.ph>
#include <APL/RSP_Protocol/RSP_Protocol.ph>
#include <APL/RTCCommon/PSAccess.h>
#include <unistd.h>
#include <string.h>
#include <blitz/array.h>
#include "BWWrite.h"
#include "Cache.h"
#include "StationSettings.h"
#define N_RETRIES 3
using namespace blitz;
using namespace LOFAR;
using namespace RSP;
using namespace EPA_Protocol;
BWWrite::BWWrite(GCFPortInterface& board_port, int board_id, int blp, int regid)
: SyncAction(board_port, board_id, MEPHeader::BF_N_FRAGMENTS),
m_blp(blp), m_regid(regid), m_remaining(0), m_offset(0)
{
memset(&m_hdr, 0, sizeof(MEPHeader));
}
BWWrite::~BWWrite()
{
}
void BWWrite::sendrequest()
{
uint8 global_blp = (getBoardId() * StationSettings::instance()->nrBlpsPerBoard()) + m_blp;
// no conditional, update every second
// reset m_offset and m_remaining for each register
if (0 == getCurrentIndex()) {
m_offset = MEPHeader::N_LOCAL_XLETS * MEPHeader::WEIGHT_SIZE;
m_remaining = MEPHeader::BF_XROUT_SIZE - m_offset; // representative for XR, XI, YR, YI size
}
if (m_regid < MEPHeader::BF_XROUT || m_regid > MEPHeader::BF_YIOUT) {
LOG_FATAL("invalid regid");
exit(EXIT_FAILURE);
}
LOG_DEBUG(formatString(">>>> BWWrite(%s) global_blp=%d, blp=%d, regid=%d, m_offset=%d, m_remaining=%d",
getBoardPort().getName().c_str(), global_blp, m_blp, m_regid, m_offset, m_remaining));
// send next BF configure message
EPABfCoefsWriteEvent bfcoefs;
size_t size = (MEPHeader::N_BEAMLETS * MEPHeader::WEIGHT_SIZE) / MEPHeader::BF_N_FRAGMENTS;
LOG_DEBUG_STR("size=" << size);
// this code is only guaranteed to work under the following conditions
ASSERT(size < MEPHeader::FRAGMENT_SIZE);
switch (m_regid) {
case MEPHeader::BF_XROUT:
bfcoefs.hdr.set(MEPHeader::BF_XROUT_HDR, 1 << m_blp, MEPHeader::WRITE, size, m_offset);
break;
case MEPHeader::BF_XIOUT:
bfcoefs.hdr.set(MEPHeader::BF_XIOUT_HDR, 1 << m_blp, MEPHeader::WRITE, size, m_offset);
break;
case MEPHeader::BF_YROUT:
bfcoefs.hdr.set(MEPHeader::BF_YROUT_HDR, 1 << m_blp, MEPHeader::WRITE, size, m_offset);
break;
case MEPHeader::BF_YIOUT:
bfcoefs.hdr.set(MEPHeader::BF_YIOUT_HDR, 1 << m_blp, MEPHeader::WRITE, size, m_offset);
break;
}
// create blitz view om the weights in the bfcoefs message to be sent to the RSP hardware
int nbeamlets_per_fragment = MEPHeader::N_BEAMLETS / MEPHeader::BF_N_FRAGMENTS;
Array<complex<int16>, 2> weights(nbeamlets_per_fragment, MEPHeader::N_POL);
bfcoefs.coef.setBuffer(weights.data(), weights.size() * sizeof(complex<uint16>));
#if 0
Array<int, 2> index(MEPHeader::N_BEAMLETS, MEPHeader::N_POL);
Array<int, 2> mapped_index(nbeamlets_per_fragment, MEPHeader::N_POL);
for (int beamlet = 0; beamlet < MEPHeader::N_BEAMLETS; beamlet++) {
for (int pol = 0; pol < MEPHeader::N_POL; pol++) {
index(beamlet, pol) = beamlet * MEPHeader::N_POL + pol;
}
}
mapped_index = 0;
#endif
LOG_DEBUG_STR("weights shape=" << weights.shape());
ASSERT(MEPHeader::N_BEAMLETS % MEPHeader::BF_N_FRAGMENTS == 0);
for (int lane = 0; lane < MEPHeader::N_SERDES_LANES; lane++) {
int hw_offset = lane;
int cache_offset = lane * (MEPHeader::N_BEAMLETS / MEPHeader::N_SERDES_LANES) + (getCurrentIndex() * nbeamlets_per_fragment / MEPHeader::N_SERDES_LANES);
Range hw_range(hw_offset, hw_offset + nbeamlets_per_fragment - MEPHeader::N_BLPS, MEPHeader::N_BLPS);
Range cache_range(cache_offset, cache_offset + (nbeamlets_per_fragment / MEPHeader::N_SERDES_LANES) - 1, 1);
LOG_DEBUG_STR("lane=" << lane);
LOG_DEBUG_STR("hw_range=" << hw_range);
LOG_DEBUG_STR("cache_range=" << cache_range);
// X
weights(hw_range, 0) = Cache::getInstance().getBack().getBeamletWeights()()(0, global_blp * 2, cache_range);
// Y
weights(hw_range, 1) = Cache::getInstance().getBack().getBeamletWeights()()(0, global_blp * 2 + 1, cache_range);
#if 0
mapped_index(hw_range, 0) = index(cache_range, 0);
mapped_index(hw_range, 1) = index(cache_range, 1);
#endif
}
#if 0
LOG_DEBUG_STR("mapped_index=" << mapped_index);
#endif
// update m_remaining and m_offset for next write
m_remaining -= size;
m_offset += size;
//
// conjugate weights to get the correct matrix
// weight a_r + i . a_i
//
// (a_r + i . a_i) . (x_r + i . x_i) = ( a_r . x_r - a_i . x_i ) + i ( a_i . x_r + a_r . x_i )
//
// This is equal to:
//
// ( x_out_r ) ( a_r -a_i ) ( x_in_r )
// ( ) = ( ) * ( )
// ( x_out_i ) ( a_i a_r ) ( x_in_i )
//
// and the same for y
//
// To send to correct weights to the beamformer we must therefore
// take the conjugate of the weight (a_r, a_i) to produce (a_r, -a_i)
// for the x_our_r component and take the conjugate AND multiply by i
// the weight (a_r, a_i) to produce (a_i, a_r).
//
weights = conj(weights);
switch (m_regid) {
case MEPHeader::BF_XROUT: {
// weights for x-real part
// no added conversions needed
// y weights should be 0
weights(Range::all(), 1) = 0;
}
break;
case MEPHeader::BF_XIOUT: {
// multiply by i to get correct weights
// weights for x-imaginary part
weights *= complex<int16>(0,1);
// y weights should be 0
weights(Range::all(), 1) = 0;
}
break;
case MEPHeader::BF_YROUT: {
// weights for y-real part
// no added conversions needed
// x weights should be 0
weights(Range::all(), 0) = 0;
}
break;
case MEPHeader::BF_YIOUT: {
// multiply by i to get correct weights
// weights for y-imaginary part
weights *= complex<int16>(0,1);
// x weights should be 0
weights(Range::all(), 0) = 0;
}
break;
}// switch
#if 0
// Debugging the waits send to one BLP by hexdumping the data before setting it on ethernet.
if (global_blp == 32 && m_offset < 1000) {
string hd;
hexdump (hd, weights.data(), weights.size() * sizeof(complex<uint16>));
LOG_INFO_STR("global_blp=32, regid="<< m_regid << ",data=" << hd << endl);
}
#endif
m_hdr = bfcoefs.hdr;
getBoardPort().send(bfcoefs);
}
void BWWrite::sendrequest_status()
{
// intentionally left empty
}
GCFEvent::TResult BWWrite::handleack(GCFEvent& event, GCFPortInterface& /*port*/)
{
if (EPA_WRITEACK != event.signal) {
LOG_WARN("BWWrite::handleack: unexpected ack");
return GCFEvent::NOT_HANDLED;
}
EPAWriteackEvent ack(event);
uint8 global_blp = (getBoardId() * StationSettings::instance()->nrBlpsPerBoard()) + m_blp;
if (!ack.hdr.isValidAck(m_hdr)) {
Cache::getInstance().getState().bf().write_error(global_blp * MEPHeader::N_PHASEPOL + m_regid);
LOG_ERROR("BWWrite::handleack: invalid ack");
return GCFEvent::NOT_HANDLED;
} else {
//
// Last fragment signals completion
//
if (MEPHeader::MEPHeader::BF_N_FRAGMENTS - 1 == getCurrentIndex()) {
Cache::getInstance().getState().bf().write_ack(global_blp * MEPHeader::N_PHASEPOL + m_regid);
}
}
return GCFEvent::HANDLED;
}