/* -*- c++ -*- */ /* * Copyright 2007 Free Software Foundation, Inc. * * This file is part of GNU Radio * * GNU Radio 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 3, or (at your option) * any later version. * * GNU Radio 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., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include typedef usrp_inband_usb_packet transport_pkt; // makes conversion to gigabit easy // FIXME We should machine generate these by a simple preprocessor run over this file // // These are all the messages that we expect to receive. // // We "intern" these here (make them into symbols) so that our // comparisions below are effectively pointer comparisons. static pmt_t s_cmd_allocate_channel = pmt_intern("cmd-allocate-channel"); static pmt_t s_cmd_close = pmt_intern("cmd-close"); static pmt_t s_cmd_deallocate_channel = pmt_intern("cmd-deallocate-channel"); static pmt_t s_cmd_open = pmt_intern("cmd-open"); static pmt_t s_cmd_start_recv_raw_samples = pmt_intern("cmd-start-recv-raw-samples"); static pmt_t s_cmd_stop_recv_raw_samples = pmt_intern("cmd-stop-recv-raw-samples"); static pmt_t s_cmd_to_control_channel = pmt_intern("cmd-to-control-channel"); static pmt_t s_cmd_xmit_raw_frame = pmt_intern("cmd-xmit-raw-frame"); static pmt_t s_cmd_max_capacity = pmt_intern("cmd-max-capacity"); static pmt_t s_cmd_ntx_chan = pmt_intern("cmd-ntx-chan"); static pmt_t s_cmd_nrx_chan = pmt_intern("cmd-nrx-chan"); static pmt_t s_cmd_current_capacity_allocation = pmt_intern("cmd-current-capacity-allocation"); static pmt_t s_response_allocate_channel = pmt_intern("response-allocate-channel"); static pmt_t s_response_close = pmt_intern("response-close"); static pmt_t s_response_deallocate_channel = pmt_intern("response-deallocate-channel"); static pmt_t s_response_from_control_channel = pmt_intern("response-from-control-channel"); static pmt_t s_response_open = pmt_intern("response-open"); static pmt_t s_response_recv_raw_samples = pmt_intern("response-recv-raw-samples"); static pmt_t s_response_xmit_raw_frame = pmt_intern("response-xmit-raw-frame"); static pmt_t s_response_max_capacity = pmt_intern("response-max-capacity"); static pmt_t s_response_ntx_chan = pmt_intern("response-ntx-chan"); static pmt_t s_response_nrx_chan = pmt_intern("response-nrx-chan"); static pmt_t s_response_current_capacity_allocation = pmt_intern("response-current-capacity-allocation"); static std::string str(long x) { std::ostringstream s; s << x; return s.str(); } usrp_server::usrp_server(mb_runtime *rt, const std::string &instance_name, pmt_t user_arg) : mb_mblock(rt, instance_name, user_arg) { // define our ports // control & status port d_cs = define_port("cs", "usrp-server-cs", true, mb_port::EXTERNAL); // ports // // (if/when we do replicated ports, these will be replaced by a // single replicated port) for(int port=0; port < N_PORTS; port++) { d_tx.push_back(define_port("tx"+str(port), "usrp-tx", true, mb_port::EXTERNAL)); d_rx.push_back(define_port("rx"+str(port), "usrp-rx", true, mb_port::EXTERNAL)); } // FIXME ... initializing to 2 channels on each for now, eventually we should // query the FPGA to get these values d_ntx_chan = 2; d_nrx_chan = 2; // Initialize capacity on each channel to 0 and to no owner for(int chan=0; chan < d_ntx_chan; chan++) { d_chaninfo_tx[chan].assigned_capacity = 0; d_chaninfo_tx[chan].owner = PMT_NIL; } for(int chan=0; chan < d_nrx_chan; chan++) { d_chaninfo_rx[chan].assigned_capacity = 0; d_chaninfo_rx[chan].owner = PMT_NIL; } } usrp_server::~usrp_server() { } void usrp_server::initial_transition() { // the initial transition } void usrp_server::handle_message(mb_message_sptr msg) { pmt_t event = msg->signal(); // the "name" of the message pmt_t port_id = msg->port_id(); // which port it came in on pmt_t data = msg->data(); pmt_t metadata = msg->metadata(); pmt_t invocation_handle; pmt_t reply_data; pmt_t status; if (1){ std::cout << "[USRP_SERVER] event: " << event << std::endl; std::cout << "[USRP_SERVER] port_id: " << port_id << std::endl; } // It would be nice if this were all table driven, and we could // compute our state transition as f(current_state, port_id, signal) if (pmt_eq(port_id, d_cs->port_symbol())){ // message came in on our control/status port if (pmt_eq(event, s_cmd_open)){ // extract args from data invocation_handle = pmt_nth(0, data); long which_usrp = pmt_to_long(pmt_nth(1, data)); // integer usrp id, usually 0 // Do the right thing.... // build a reply (void) which_usrp; // avoid unused warning // if everything OK status = PMT_T; reply_data = pmt_list2(invocation_handle, status); // ...and send it d_cs->send(s_response_open, reply_data); return; } else if (pmt_eq(event, s_cmd_close)){ // ... } else if (pmt_eq(event, s_cmd_max_capacity)) { invocation_handle = pmt_nth(0, data); reply_data = pmt_list2(invocation_handle, pmt_from_long(max_capacity())); d_cs->send(s_response_max_capacity, reply_data); return; } else if (pmt_eq(event, s_cmd_ntx_chan)) { invocation_handle = pmt_nth(0, data); reply_data = pmt_list2(invocation_handle, pmt_from_long(d_ntx_chan)); d_cs->send(s_response_ntx_chan, reply_data); } else if (pmt_eq(event, s_cmd_nrx_chan)) { invocation_handle = pmt_nth(0, data); reply_data = pmt_list2(invocation_handle, pmt_from_long(d_nrx_chan)); d_cs->send(s_response_nrx_chan, reply_data); } else if (pmt_eq(event, s_cmd_current_capacity_allocation)) { invocation_handle = pmt_nth(0, data); reply_data = pmt_list2(invocation_handle, pmt_from_long(current_capacity_allocation())); d_cs->send(s_response_current_capacity_allocation, reply_data); } goto unhandled; } if (pmt_eq(event, s_cmd_allocate_channel)){ handle_cmd_allocate_channel(port_id, data); return; } if (pmt_eq(event, s_cmd_deallocate_channel)) { handle_cmd_deallocate_channel(port_id, data); return; } if (pmt_eq(event, s_cmd_xmit_raw_frame)){ handle_cmd_xmit_raw_frame(data); return; } unhandled: std::cout << "[USRP_SERVER] unhandled msg: " << msg << std::endl; } // Return -1 if it is not an RX port, or an index int usrp_server::tx_port_index(pmt_t port_id) { for(int i=0; i < (int) d_tx.size(); i++) if(pmt_eq(d_tx[i]->port_symbol(), port_id)) return i; return -1; } // Return -1 if it is not an RX port, or an index int usrp_server::rx_port_index(pmt_t port_id) { for(int i=0; i < (int) d_rx.size(); i++) if(pmt_eq(d_rx[i]->port_symbol(), port_id)) return i; return -1; } // Go through all TX and RX channels, sum up the assigned capacity // and return it long usrp_server::current_capacity_allocation() { long capacity = 0; for(int chan=0; chan < d_ntx_chan; chan++) capacity += d_chaninfo_tx[chan].assigned_capacity; for(int chan=0; chan < d_nrx_chan; chan++) capacity += d_chaninfo_rx[chan].assigned_capacity; return capacity; } void usrp_server::handle_cmd_allocate_channel(pmt_t port_id, pmt_t data) { pmt_t invocation_handle = pmt_nth(0, data); long rqstd_capacity = pmt_to_long(pmt_nth(1, data)); long chan, port; pmt_t reply_data; // If it's a TX port, allocate on a free channel, else check if it's a RX port // and allocate. if((port = tx_port_index(port_id)) != -1) { // Check capacity exists if((D_USB_CAPACITY - current_capacity_allocation()) < rqstd_capacity) { reply_data = pmt_list3(invocation_handle, pmt_from_long(RQSTD_CAPACITY_UNAVAIL), PMT_NIL); // no capacity available d_tx[port]->send(s_response_allocate_channel, reply_data); return; } // Find a free channel, assign the capacity and respond for(chan=0; chan < d_ntx_chan; chan++) { if(d_chaninfo_tx[chan].owner == PMT_NIL) { d_chaninfo_tx[chan].owner = port_id; d_chaninfo_tx[chan].assigned_capacity = rqstd_capacity; reply_data = pmt_list3(invocation_handle, PMT_T, pmt_from_long(chan)); d_tx[port]->send(s_response_allocate_channel, reply_data); return; } } std::cout << "[USRP_SERVER] Couldnt find a TX chan\n"; reply_data = pmt_list3(invocation_handle, pmt_from_long(CHANNEL_UNAVAIL), PMT_NIL); // no free TX chan found d_tx[port]->send(s_response_allocate_channel, reply_data); return; } // Repeat the same process on the RX side if the port was not determined to be TX if((port = rx_port_index(port_id)) != -1) { if((D_USB_CAPACITY - current_capacity_allocation()) < rqstd_capacity) { reply_data = pmt_list3(invocation_handle, pmt_from_long(RQSTD_CAPACITY_UNAVAIL), PMT_NIL); // no capacity available d_rx[port]->send(s_response_allocate_channel, reply_data); return; } for(chan=0; chan < d_nrx_chan; chan++) { if(d_chaninfo_rx[chan].owner == PMT_NIL) { d_chaninfo_rx[chan].owner = port_id; d_chaninfo_rx[chan].assigned_capacity = rqstd_capacity; reply_data = pmt_list3(invocation_handle, PMT_T, pmt_from_long(chan)); d_rx[port]->send(s_response_allocate_channel, reply_data); return; } } std::cout << "[USRP_SERVER] Couldnt find a RX chan\n"; reply_data = pmt_list3(invocation_handle, pmt_from_long(CHANNEL_UNAVAIL), PMT_NIL); // no free RX chan found d_rx[port]->send(s_response_allocate_channel, reply_data); return; } } // Check the port type and deallocate assigned capacity based on this, ensuring // that the owner of the method invocation is the owner of the port and that // the channel number is valid. void usrp_server::handle_cmd_deallocate_channel(pmt_t port_id, pmt_t data) { pmt_t invocation_handle = pmt_nth(0, data); long channel = pmt_to_long(pmt_nth(1, data)); long port; pmt_t reply_data; // Check that the channel number is valid, and that the calling port is the owner // of the channel, and if so remove the assigned capacity. if((port = tx_port_index(port_id)) != -1) { if(channel >= d_ntx_chan) { reply_data = pmt_list2(invocation_handle, pmt_from_long(CHANNEL_INVALID)); // not a legit channel number d_tx[port]->send(s_response_deallocate_channel, reply_data); return; } if(d_chaninfo_tx[channel].owner != port_id) { reply_data = pmt_list2(invocation_handle, pmt_from_long(PERMISSION_DENIED)); // not the owner of the port d_tx[port]->send(s_response_deallocate_channel, reply_data); return; } d_chaninfo_tx[channel].assigned_capacity = 0; d_chaninfo_tx[channel].owner = PMT_NIL; reply_data = pmt_list2(invocation_handle, PMT_T); d_tx[port]->send(s_response_deallocate_channel, reply_data); return; } // Repeated process on the RX side if((port = rx_port_index(port_id)) != -1) { if(channel >= d_nrx_chan) { reply_data = pmt_list2(invocation_handle, pmt_from_long(CHANNEL_INVALID)); // not a legit channel number d_rx[port]->send(s_response_deallocate_channel, reply_data); return; } if(d_chaninfo_rx[channel].owner != port_id) { reply_data = pmt_list2(invocation_handle, pmt_from_long(PERMISSION_DENIED)); // not the owner of the port d_rx[port]->send(s_response_deallocate_channel, reply_data); return; } d_chaninfo_rx[channel].assigned_capacity = 0; d_chaninfo_rx[channel].owner = PMT_NIL; reply_data = pmt_list2(invocation_handle, PMT_T); d_rx[port]->send(s_response_deallocate_channel, reply_data); return; } } void usrp_server::handle_cmd_xmit_raw_frame(pmt_t data) { size_t n_bytes, psize; long max_payload_len = transport_pkt::max_payload(); pmt_t invocation_handle = pmt_nth(0, data); long channel = pmt_to_long(pmt_nth(1, data)); const void *samples = pmt_uniform_vector_elements(pmt_nth(2, data), n_bytes); long timestamp = pmt_to_long(pmt_nth(3, data)); // Determine the number of packets to allocate contiguous memory for bursting over the // USB and get a pointer to the memory to be used in building the packets long n_packets = static_cast(std::ceil(n_bytes / (double)max_payload_len)); pmt_t v_packets = pmt_make_u8vector(sizeof(transport_pkt) * n_packets, 0); transport_pkt *pkts = (transport_pkt *) pmt_u8vector_writeable_elements(v_packets, psize); for(int n=0; n < n_packets; n++) { long payload_len = std::min((long)(n_bytes-(n*max_payload_len)), (long)max_payload_len); if(n == 0) { // first packet gets start of burst flag and timestamp pkts[n].set_header(pkts[n].FL_START_OF_BURST, channel, 0, payload_len); pkts[n].set_timestamp(timestamp); } else { pkts[n].set_header(0, channel, 0, payload_len); pkts[n].set_timestamp(0xffffffff); } memcpy(pkts[n].payload(), (uint8_t *)samples+(max_payload_len * n), payload_len); } pkts[n_packets-1].set_end_of_burst(); // set the last packet's end of burst // interface with the USRP to send the USB packet, since the memory is // contiguous, this should be a serious of memory copies to the bus, each being // USB_PKT_SIZE * MAX_PACKET_BURST bytes worth of data (given a full burst) } REGISTER_MBLOCK_CLASS(usrp_server);