/* -*- c++ -*- */ /* * Copyright 2007,2008 Free Software Foundation, Inc. * * 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 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 . */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "app_passthru_v2.h" #include "buffer_pool.h" #include "memcpy_wa.h" #include "ethernet.h" #include "nonstdio.h" #include "print_rmon_regs.h" #include "db.h" #include "clocks.h" #include volatile bool link_is_up = false; // eth handler sets this // If this is non-zero, this dbsm could be writing to the ethernet dbsm_t *ac_could_be_sending_to_eth; //static unsigned char exp_seqno = 0; void set_reply_hdr(u2_eth_packet_t *reply_pkt, u2_eth_packet_t const *cmd_pkt) { reply_pkt->ehdr.dst = cmd_pkt->ehdr.src; reply_pkt->ehdr.src = *ethernet_mac_addr(); reply_pkt->ehdr.ethertype = U2_ETHERTYPE; reply_pkt->thdr.flags = 0; reply_pkt->thdr.fifo_status = 0; // written by protocol engine reply_pkt->thdr.seqno = 0; // written by protocol engine reply_pkt->thdr.ack = 0; // written by protocol engine u2p_set_word0(&reply_pkt->fixed, 0, CONTROL_CHAN); reply_pkt->fixed.timestamp = timer_regs->time; } static void send_reply(unsigned char *reply, size_t reply_len) { if (reply_len < 64) reply_len = 64; // wait for buffer to become idle hal_set_leds(0x4, 0x4); while((buffer_pool_status->status & BPS_IDLE(CPU_TX_BUF)) == 0) ; hal_set_leds(0x0, 0x4); // copy reply into CPU_TX_BUF memcpy_wa(buffer_ram(CPU_TX_BUF), reply, reply_len); // wait until nobody else is sending to the ethernet if (ac_could_be_sending_to_eth){ hal_set_leds(0x8, 0x8); dbsm_wait_for_opening(ac_could_be_sending_to_eth); hal_set_leds(0x0, 0x8); } // fire it off bp_send_from_buf(CPU_TX_BUF, PORT_ETH, 1, 0, reply_len/4); // wait for it to complete (not long, it's a small pkt) while((buffer_pool_status->status & (BPS_DONE(CPU_TX_BUF) | BPS_ERROR(CPU_TX_BUF))) == 0) ; bp_clear_buf(CPU_TX_BUF); } static size_t op_id_cmd(const op_generic_t *p, void *reply_payload, size_t reply_payload_space) { op_id_reply_t *r = (op_id_reply_t *) reply_payload; if (reply_payload_space < sizeof(*r)) // no room return 0; // Build reply subpacket r->opcode = OP_ID_REPLY; r->len = sizeof(op_id_reply_t); r->rid = p->rid; r->addr = *ethernet_mac_addr(); r->hw_rev = 0x0000; // FIXME // r->fpga_md5sum = ; // FIXME // r->sw_md5sum = ; // FIXME // FIXME Add d'board info, including dbid, min/max gain, min/max freq return r->len; } static size_t add_eop(void *reply_payload, size_t reply_payload_space) { op_generic_t *r = (op_generic_t *) reply_payload; if (reply_payload_space < sizeof(*r)) return 0; // no room r->opcode = OP_EOP; r->len = sizeof(*r); r->rid = 0; r->ok = 0; return r->len; } bool handle_control_chan_frame(u2_eth_packet_t *pkt, size_t len) { unsigned char reply[sizeof(u2_eth_packet_t) + 4 * sizeof(u2_subpkt_t)] _AL4; unsigned char *reply_payload = &reply[sizeof(u2_eth_packet_t)]; int reply_payload_space = sizeof(reply) - sizeof(u2_eth_packet_t); bool handled_it = false; // initialize reply memset(reply, 0, sizeof(reply)); set_reply_hdr((u2_eth_packet_t *) reply, pkt); // point to beginning of payload (subpackets) unsigned char *payload = ((unsigned char *) pkt) + sizeof(u2_eth_packet_t); int payload_len = len - sizeof(u2_eth_packet_t); size_t subpktlen = 0; while (payload_len >= sizeof(op_generic_t)){ const op_generic_t *gp = (const op_generic_t *) payload; subpktlen = 0; switch(gp->opcode){ case OP_EOP: // end of subpackets goto end_of_subpackets; case OP_ID: subpktlen = op_id_cmd(gp, reply_payload, reply_payload_space); handled_it = true; break; default: if (0){ printf("\npassing on %d\n", gp->opcode); } break; } int t = (gp->len + 3) & ~3; // bump to a multiple of 4 payload += t; payload_len -= t; subpktlen = (subpktlen + 3) & ~3; // bump to a multiple of 4 reply_payload += subpktlen; reply_payload_space -= subpktlen; } end_of_subpackets: if (handled_it){ // add the EOP marker subpktlen = add_eop(reply_payload, reply_payload_space); subpktlen = (subpktlen + 3) & ~3; // bump to a multiple of 4 reply_payload += subpktlen; reply_payload_space -= subpktlen; send_reply(reply, reply_payload - reply); } return handled_it; } /* * Called when an ethernet packet is received. * Return true if we handled it here, otherwise * it'll be passed on to the DSP Tx pipe */ bool eth_pkt_inspector(dbsm_t *sm, int bufno) { u2_eth_packet_t *pkt = (u2_eth_packet_t *) buffer_ram(bufno); size_t byte_len = (buffer_pool_status->last_line[bufno] - 3) * 4; //static size_t last_len = 0; // hal_toggle_leds(0x1); // inspect rcvd frame and figure out what do do. if (pkt->ehdr.ethertype != U2_ETHERTYPE) return true; // ignore, probably bogus PAUSE frame from MAC int chan = u2p_chan(&pkt->fixed); switch (chan){ case CONTROL_CHAN: return handle_control_chan_frame(pkt, byte_len); break; case 0: default: #if 0 if (last_len != 0){ if (byte_len != last_len){ printf("Len: %d last: %d\n", byte_len, last_len); } } last_len = byte_len; if((pkt->thdr.seqno) == exp_seqno){ exp_seqno++; //putchar('.'); } else { // putchar('S'); //printf("S%d %d ",exp_seqno,pkt->thdr.seqno); exp_seqno = pkt->thdr.seqno + 1; } #endif return false; // pass it on to Tx DSP break; } } /* * Called when eth phy state changes (w/ interrupts disabled) */ void link_changed_callback(int speed) { link_is_up = speed != 0; hal_set_leds(link_is_up ? LED_RJ45 : 0x0, LED_RJ45); printf("\neth link changed: speed = %d\n", speed); }