/* * Copyright 2010 Stefan Lankes, Chair for Operating Systems, * RWTH Aachen University * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR COND */ #include #include #include #include #include #include #include #include #ifdef CONFIG_ROCKCREEK bootinfo_t* bootinfo = (bootinfo_t*) SCC_BOOTINFO; /* PSE bit for Pentium+ equals MPE (message buffer enable) flag in RCK! So, use it to create _PAGE_MPB symbol... */ #define _CR4_MPE 0x00000800 /* * This is the modified MPB program, which is part of the RCCE distribution (src/mpb.c). * * This function clears the local MPB and resets the test&set register. */ static int scc_clear(void) { int tmp, x, y, z, offset; // Initialize API InitAPI(0); // Find out who I am... tmp=ReadConfigReg(CRB_OWN+MYTILEID); x=(tmp>>3) & 0x0f; // bits 06:03 y=(tmp>>7) & 0x0f; // bits 10:07 z=(tmp ) & 0x07; // bits 02:00 // Allocate Message Passing Buffer t_vcharp MPB; MPBalloc(&MPB, x, y, z, 1); if (!MPB) { kprintf("Unable to allocate MPB for core %d of Tile x=%d, y= %d! Exiting.\n", z, x, y); return 255; } // zap own MPB for (offset=0; offset < 0x2000; offset+=8) *(volatile unsigned long long int*)(MPB+offset) = 0; // Clear test&set register write. Next read-access will read "1" (lock granted). SetConfigReg(CRB_ADDR(x,y)+((z)?LOCK1:LOCK0), 1); // frees Message Passing Buffer MPBunalloc(&MPB); return 0; } int scc_init(void) { int num_ranks; int i, my_rank; uint64_t start, end, ticks, freq = 533; uint32_t cr4, msg = 0; kputs("Initialize Rock Creek!\n"); /* Enable Messagepassing in CR4 */ cr4 = read_cr4(); cr4 = cr4 | _CR4_MPE; write_cr4(cr4); kprintf("address of the initrd: 0x%x\n", bootinfo->addr); kprintf("size of the initrd: %d\n", bootinfo->size); kprintf("rcce argc = %d\n", bootinfo->argc); for(i=0; iargc; i++) kprintf("rcce argv[%d] = %s\n", i, bootinfo->argv[i]); if (bootinfo->argc >= 3) freq = atoi(bootinfo->argv[2]); kputs("Reset SCC!\n"); scc_clear(); kputs("Wait some time...\n"); mb(); start = rdtsc(); do { mb(); end = rdtsc(); ticks = end > start ? end - start : start - end; } while(ticks*TIMER_FREQ < 300ULL*freq*1000000ULL); if (RCCE_init(&bootinfo->argc, &bootinfo->argv) != RCCE_SUCCESS) return -ENODEV; if (iRCCE_init() != iRCCE_SUCCESS) return -ENODEV; // enable additional outputs RCCE_debug_set(RCCE_DEBUG_ALL); my_rank = RCCE_ue(); num_ranks = RCCE_num_ues(); kprintf("Got rank %d of %d ranks\n", my_rank, num_ranks); i = ReadConfigReg(CRB_OWN+GLCFG0); kprintf("glcfg0 0x%x\n", i); RCCE_barrier(&RCCE_COMM_WORLD); kputs("RCCE test...\t"); if (my_rank == 0) msg = 0x4711; if (RCCE_bcast((char*) &msg, sizeof(msg), 0, RCCE_COMM_WORLD) == RCCE_SUCCESS) kprintf("successfull! (0x%x)\n", msg); else kprintf("failed! (0x%x)\n", msg); kputs("Now, the SCC is initialized!\n"); return 0; } #endif