Skip to content
Snippets Groups Projects
wrc_main.c 4.36 KiB
Newer Older
/*
 * This work is part of the White Rabbit project
 *
 * Copyright (C) 2011,2012 CERN (www.cern.ch)
 * Author: Tomasz Wlostowski <tomasz.wlostowski@cern.ch>
 * Author: Grzegorz Daniluk <grzegorz.daniluk@cern.ch>
 *
 * Released according to the GNU GPL, version 2 or any later version.
 */
#include <stdio.h>
#include <inttypes.h>

#include "syscon.h"
#include "uart.h"
#include "endpoint.h"
#include "minic.h"
#include "pps_gen.h"
#include "ptpd_netif.h"
Grzegorz Daniluk's avatar
Grzegorz Daniluk committed
#include "i2c.h"
#include "softpll_ng.h"
#include "pps_gen.h"
Wesley W. Terpstra's avatar
Wesley W. Terpstra committed
#include "lib/ipv4.h"
#include "rxts_calibrator.h"
Grzegorz Daniluk's avatar
Grzegorz Daniluk committed

#include "system_checks.h"
int wrc_phase_tracking = 1;
///////////////////////////////////
//Calibration data (from EEPROM if available)
int32_t sfp_alpha = 73622176;	//default values if could not read EEPROM
int32_t sfp_deltaTx = 0;
int32_t sfp_deltaRx = 0;
uint32_t cal_phase_transition = 2389;
static void wrc_initialize(void)
	uint8_t mac_addr[6];
	sdb_find_devices();
	uart_init_sw();
	uart_init_hw();
	pp_printf("WR Core: starting up...\n");
	wrpc_w1_bus.detail = ONEWIRE_PORT;
	w1_scan_bus(&wrpc_w1_bus);
	/*initialize I2C bus*/
	mi2c_init(WRPC_FMC_I2C);
	/*init storage (Flash / W1 EEPROM / I2C EEPROM*/
	storage_init(WRPC_FMC_I2C, FMC_EEPROM_ADR);
	if (get_persistent_mac(ONEWIRE_PORT, mac_addr) == -1) {
		pp_printf("Unable to determine MAC address\n");
		mac_addr[0] = 0x22;	//
		mac_addr[1] = 0x33;	//
		mac_addr[2] = 0x44;	// fallback MAC if get_persistent_mac fails
		mac_addr[3] = 0x55;	//
		mac_addr[4] = 0x66;	//
		mac_addr[5] = 0x77;	//
	pp_printf("Local MAC address: %02x:%02x:%02x:%02x:%02x:%02x\n",
		mac_addr[0], mac_addr[1], mac_addr[2], mac_addr[3],
		mac_addr[4], mac_addr[5]);
	ep_init(mac_addr);
	ep_enable(1, 1);
	shw_pps_gen_init();
	//try reading t24 phase transition from EEPROM
	calib_t24p(WRC_MODE_MASTER, &cal_phase_transition);
	if (HAS_IP) {
		ipv4_init();
		arp_init();
	}
#define LINK_WENT_UP 1
#define LINK_WENT_DOWN 2
#define LINK_UP 3
#define LINK_DOWN 4
static int wrc_check_link(void)
	int link_state = ep_link_up(NULL);
	if (!prev_link_state && link_state) {
		wrc_verbose("Link up.\n");
		gpio_out(GPIO_LED_LINK, 1);
		rv = LINK_WENT_UP;
	} else if (prev_link_state && !link_state) {
		wrc_verbose("Link down.\n");
		gpio_out(GPIO_LED_LINK, 0);
		rv = LINK_WENT_DOWN;
	} else
		rv = (link_state ? LINK_UP : LINK_DOWN);
	prev_link_state = link_state;

	return rv;
static void ui_update(void)
	if (wrc_ui_mode == UI_GUI_MODE) {
		wrc_mon_gui();
		if (uart_read_byte() == 27 || wrc_ui_refperiod == 0) {
			shell_init();
			wrc_ui_mode = UI_SHELL_MODE;
		shell_interactive();
	}
	/* Stats is asynchronous now. It's not a different mode, but a flag */
	if (wrc_stat_running)
		wrc_log_stats();
/* initialize functions to be called after reset in check_reset function */
void init_hw_after_reset(void)
	/* Ok, now init the devices so we can printf and delay */
	sdb_find_devices();
	extern uint32_t uptime_sec;
	uint32_t j, lastj, fraction = 0;

	wrc_ui_mode = UI_SHELL_MODE;
	_endram = ENDRAM_MAGIC;
	wrc_ptp_set_mode(WRC_MODE_SLAVE);
	wrc_ptp_start();
	//try to read and execute init script from EEPROM
	shell_boot_script();
	for (;;) {
		int l_status = wrc_check_link();
		/* count uptime, in seconds, for remote polling */
		j = timer_get_tics();
		fraction += j -lastj;
		lastj = j;
		while (fraction > TICS_PER_SECOND) {
			fraction -= TICS_PER_SECOND;
			uptime_sec++;
		}

		switch (l_status) {
		case LINK_WENT_UP:
			needIP = 1;
			break;
		case LINK_UP:
			update_rx_queues();
			if (HAS_IP) {
				ipv4_poll();
				arp_poll();
			}
			break;

		case LINK_WENT_DOWN:
			if (wrc_ptp_get_mode() == WRC_MODE_SLAVE) {
				spll_init(SPLL_MODE_FREE_RUNNING_MASTER, 0, 1);
				shw_pps_gen_enable_output(0);
			}
			break;
		}

		ui_update();
		wrc_ptp_update();
		check_stack();