view TaskManager/Cell/CellTaskManagerImpl.cc @ 812:a939e4d6fe5f draft

fix Cell
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Sun, 23 May 2010 00:23:46 +0900
parents 13c736659c63
children af2fb2e641eb
line wrap: on
line source

#define DEBUG
#include "error.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "CellTaskManagerImpl.h"
#include "HTaskInfo.h"
#include "SchedTask.h"
#include "MainScheduler.h"
#include "types.h"
#include "SysFunc.h"

static void send_alloc_reply(CellTaskManagerImpl *tm, int id,  SpeThreads *speThreads);

CellTaskManagerImpl::~CellTaskManagerImpl()
{
    delete speThreads;
    delete [] speTaskList;

    delete ppeManager;
}

void
CellTaskManagerImpl::init()
{
    spe_running = 0;

    activeTaskQueue = new HTaskInfo();

    htaskImpl = activeTaskQueue ; // any HTaskInfo

    speThreads = new SpeThreads(machineNum);
    speThreads->init();

    speTaskList  = new TaskListInfoPtr[machineNum];
    taskListInfo  = new TaskListInfoPtr[machineNum];

    for (int i = 0; i < machineNum; i++) {
	taskListInfo[i] = new TaskListInfo();
	speTaskList[i] = new TaskListInfo();
    }

    // PPE 側の管理をする Manager
    ppeManager = new FifoTaskManagerImpl(machineNum);
    // 大半のTaskQueueInfoは、共有される
    MainScheduler *mscheduler = new MainScheduler;
    ppeManager->init(mscheduler, this);
    
    ppeManager->get_scheduler()->set_manager(this);

    schedTaskManager = new SchedTask();
    schedTaskManager->init(0,0,0,ppeManager->get_scheduler());
}

void
CellTaskManagerImpl::append_activeTask(HTaskPtr task)
{
    if (task->cpu_type == CPU_PPE) {
        ppeManager->append_activeTask(task);
    } else {
        activeTaskQueue->addLast(task);
    }
}

// SPE_ANY が指定されていた時に
// これをインクリメントしつつ呼ぶことにする。
// 乱数使ってもいいけどさ。
int cur_anySpeid = 0;

/**
 * ActiveTaskQueue から Task を
 * 各 SPE に渡す TaskList に入れる
 *
 * ここの activeTaskQueue は FifoTaskManagerImpl のと意味が違い、
 * spe に渡される Task だけ入っている
 */
void
CellTaskManagerImpl::set_runTaskList()
{
    int speid;

    while (HTaskPtr htask = activeTaskQueue->poll()) {

	if (htask->cpu_type == SPE_ANY) {
	    speid = cur_anySpeid++;
	    cur_anySpeid = (cur_anySpeid < machineNum)
		? cur_anySpeid : 0;
	} else {
	    // -1 してるのは
	    // htask->cpu_type - CPU_SPE で
	    // SPE0 = 1, SPE1 = 2, ... SPE5 = 6 ってなってるので
	    // 配列的 (SPE0 = arr[0], SPE1 = arr[1]) にするため
	    speid = htask->cpu_type - CPU_SPE - 1;

	    // SPU の数以上が指定されていれば
	    // とりあえず MAX_USE_SPE_NUM (実際に動く SPE の最大数) で
	    // あまり求めてそれを使うことにする。
	    // ここで判定するもんでもないか?
	    if (speid >= machineNum) {
		speid %= machineNum;
	    }
	}
	set_taskList(htask, taskListInfo[speid]);
    }
}

void
CellTaskManagerImpl::sendTaskList()
{
    for (int i = 0; i < machineNum; i++)  {
	if ( taskListInfo[i]->length() > 0 ) {
	    send_taskList(i);
	    spe_running++;
	}
    }
}

void
CellTaskManagerImpl::poll()
{
    mail_check();
    // SPE に送る TaskList の準備
    set_runTaskList();
    // TaskList 待ちの SPE に TaskList を送る
    sendTaskList();
}

void
CellTaskManagerImpl::run()
{
    do {
        // PPE side
	ppeManager->poll();
        // SPE side
	do {
	    poll();
	} while (ppeManager->activeTaskQueue->empty() && spe_running >0 );
    } while (!ppeManager->taskListInfo->empty() || spe_running >0); 
    if (!waitTaskQueue->empty()) {
	get_scheduler()->printf("Dead lock detected\n");
    }
}

/**
 * SPE からのメールをチェックする
 */

void
CellTaskManagerImpl::mail_check()
{
    memaddr data;

    // SPE Scheduler からの mail check
    for (int id = 0; id < machineNum; id++) {	    
	while (speThreads->has_mail(id, 1, &data)) {				
	    if (data == (memaddr)MY_SPE_STATUS_READY) {
		//  MY_SPE_STATUS_READY: SPE が持ってた Task 全て終了
	        speTaskList[id]->freeAll();
		spe_running--;
	    } else if (data == (memaddr)MY_SPE_COMMAND_MALLOC) {
	        // MY_SPE_COMMAND_MALLOC   SPE からのmain memory request
		send_alloc_reply(this, id, speThreads);
	    } else if (data > (memaddr)MY_SPE_NOP) {
	        // 終了したタスク(PPEにあるのでアドレス)
		HTaskPtr task = (HTaskPtr)data;
		task->post_func(schedTaskManager, task->post_arg1, task->post_arg2);
		check_task_finish(task,waitTaskQueue);
	    }
	    // MY_SPE_NOP: 特に意味のないコマンド
	}
    }
}

static void
send_alloc_reply(CellTaskManagerImpl *tm, int id, SpeThreads *speThreads)
{

    /**
     * info[0] = alloc_id; (CellScheduler::mainMem_alloc 参照)
     * info[1] = alloc_addr;
     */
    memaddr alloc_info[2];
    long alloc_size;
    long command;
    
    speThreads->get_mail(id, 2, alloc_info);
    command = (long)alloc_info[0];
    alloc_size = (long)alloc_info[1];

    
    alloc_info[1] = (memaddr)tm->allocate(alloc_size);
    //__debug_ppe("[PPE] MALLOCED 0x%lx from [SPE %d]\n", alloc_info[1],id);
    // 今のところ何もしてない。どうも、この allocate を free 
    // するのは、SPE task が返した値を見て行うらしい。それは、
    // 忘れやすいのではないか?
    speThreads->add_output_tasklist(command, alloc_info[1], alloc_size);

    speThreads->send_mail(id, 2, alloc_info);
}

/**
 * 条件を満たしたら SPE に TaskList を送信する
 * 条件1. SPE が持ってた TaskList を終了して、次の TaskList を待ってる
 * 条件2. SPE に送る TaskList に Task がある
 *
 * SPE で実行終了した speTaskList  と
 * これから実行する taskListInfo  のバッファを入れ替える
 */
void
CellTaskManagerImpl::send_taskList(int id)
{
    if (taskListInfo[id]->empty()) return;
    TaskListInfoPtr tmp = taskListInfo[id];
    taskListInfo[id] = speTaskList[id];
    speTaskList[id] = tmp;

    tmp->getLast()->next = 0;
    TaskListPtr p = tmp->getFirst();
    speThreads->send_mail(id, 1, (memaddr *)&p);
}

void CellTaskManagerImpl::show_profile() {
    for (int id = 0; id < machineNum; id++) {	    
	HTaskPtr t = create_task(ShowTime);
	t->set_cpu((CPU_TYPE)(id+2));
	t->spawn();
    }
}

void CellTaskManagerImpl::start_profile() {
    for (int id = 0; id < machineNum; id++) {	    
	HTaskPtr t = create_task(StartProfile);
	t->set_cpu((CPU_TYPE)(id+2));
	t->spawn();
    }
}


#ifdef __CERIUM_CELL__
TaskManagerImpl*
create_impl(int num)
{
    return new CellTaskManagerImpl(num);
}
#endif // __CERIUM_CELL