All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.sun.electric.tool.MultiTaskJob Maven / Gradle / Ivy

/* -*- tab-width: 4 -*-
 *
 * Electric(tm) VLSI Design System
 *
 * File: MultiTaskJob.java
 *
 * Copyright (c) 2003, Oracle and/or its affiliates. All rights reserved.
 *
 * Electric(tm) 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.
 *
 * Electric(tm) 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 Electric(tm); see the file COPYING.  If not, write to
 * the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
 * Boston, Mass 02111-1307, USA.
 */
package com.sun.electric.tool;

import com.sun.electric.database.Environment;
import java.lang.management.ManagementFactory;
import java.lang.management.ThreadMXBean;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.Map;

/**
 * This generic class supports map-reduce scheme of computation on Electric database.
 * Large computation has three stages:
 * 1) Large computation is splitted into smaller tasks.
 * Smaller tasks are identified by TaskKey class.
 * This stage is performed by prepareTasks method, which schedules each task by startTask method.
 * 2) Tasks run in parallel, each giving result of TaskResult type.
 * This stage is performed by runTask method for each instance of task.
 * 3) TaskResults are combinded into final result of Result type.
 * This stage is performed by mergeTaskResults method.
 * 4) Result is consumed on server.
 * This stage is performed by consumer.consume method.
 */
public abstract class MultiTaskJob extends Job {

    private transient LinkedHashMap tasks;
    private transient ArrayList allTasks;
    private transient int tasksStarted;
    private transient int tasksDone;
    private transient Environment env;
    private transient EThread ownerThread;
    private transient int numberOfRunningThreads;
    private transient int numberOfFinishedThreads;
    private Consumer consumer;

    private transient ThreadMXBean threadMX;
    private transient long accumulatedCpuTime;
    private transient long accumulatedUserTime;

    private static final double MILLIS_IN_SEC = 1e3;
    private static final double NANOS_IN_SEC = 1e9;

    /**
     * Constructor creates a new instance of MultiTaskJob.
     * @param jobName a string that describes this MultiTaskJob.
     * @param t the Tool that originated this MultiTaskJob.
     * @param c interface which consumes the result on server
     */
    public MultiTaskJob(String jobName, Tool t, Consumer c) {
        super(jobName, t, Job.Type.SERVER_EXAMINE, null, null, Job.Priority.USER);
        this.consumer = c;
    }

    /**
     * This abstract method split large computation into smaller task.
     * Smaller tasks are identified by TaskKey class.
     * Each task is scheduled by startTask method.
     * @throws com.sun.electric.tool.JobException
     */
    public abstract void prepareTasks() throws JobException;

    /**
     * This abtract methods performs computation of each task.
     * @param taskKey task key which identifies the task
     * @return result of task computation
     * @throws com.sun.electric.tool.JobException
     */
    public abstract TaskResult runTask(TaskKey taskKey) throws JobException;

    /**
     * This abtract method combines task results into final result.
     * @param taskResults map which contains result of each completed task.
     * @return final result which is obtained by merging task results.
     * @throws com.sun.electric.tool.JobException
     */
    public abstract Result mergeTaskResults(Map taskResults) throws JobException;

//    /**
//     * This method executes in the Client side after normal termination of full computation.
//     * This method should perform all needed termination actions.
//     * @param result result of full computation.
//     */
//    public void terminateOK(Result result) {}
    /**
     * This method is not overriden by subclasses.
     * Override methods prepareTasks, runTask, mergeTaskResults instead.
     * @throws JobException
     */
    @Override
    public final boolean doIt() throws JobException {
        threadMX = ManagementFactory.getThreadMXBean();
        long startClockTime = System.currentTimeMillis();
        long startCpuTime = threadMX.getCurrentThreadCpuTime();
        long startUserTime = threadMX.getCurrentThreadUserTime();
        tasks = new LinkedHashMap();
        allTasks = new ArrayList();
        prepareTasks();
        env = Environment.getThreadEnvironment();
        ownerThread = (EThread) Thread.currentThread();
        numberOfRunningThreads = ServerJobManager.getMaxNumberOfThreads();
        tasksDone = -numberOfRunningThreads;
        for (int id = 0; id < numberOfRunningThreads; id++) {
            new MultiTaskWorkingThread(id).start();
        }
        waitTasks();

        LinkedHashMap taskResults = new LinkedHashMap();
        for (Task task : tasks.values()) {
            if (task.taskResult != null) {
                taskResults.put(task.taskKey, task.taskResult);
            }
        }
        tasks.clear();
        Result result = mergeTaskResults(taskResults);
        taskResults.clear();
        long endClockTime = System.currentTimeMillis();
        accumulatedCpuTime += (threadMX.getCurrentThreadCpuTime() - startCpuTime);
        accumulatedUserTime += (threadMX.getCurrentThreadCpuTime() - startUserTime);
        System.out.println(this  + " took " +
                (endClockTime - startClockTime)/MILLIS_IN_SEC + " sec, cpu=" + accumulatedCpuTime/NANOS_IN_SEC + " user=" + accumulatedUserTime/NANOS_IN_SEC);
        if (consumer != null) {
            consumer.consume(result);
        }
        return true;
    }

    /**
     * Schedules task. Should be callled from prepareTasks or runTask methods only.
     * @param taskName task name which is appeared in Jobs Explorer Tree
     * @param taskKey task key which identifies the task.
     */
    public synchronized void startTask(String taskName, TaskKey taskKey) {
        Task task = new Task(taskName, taskKey);
        if (tasks.containsKey(taskKey)) {
            throw new IllegalArgumentException();
        }
        tasks.put(taskKey, task);
        allTasks.add(task);
        notifyAll();
    }

    private synchronized Task getTask() {
        tasksDone++;
        assert tasksDone <= tasksStarted;
        try {
            while (tasksDone < allTasks.size() && tasksStarted == allTasks.size()) {
                wait();
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        if (tasksDone == allTasks.size()) {
            assert tasksStarted == tasksDone;
            return null;
        }
        return allTasks.get(tasksStarted++);
    }

    private synchronized void waitTasks() {
        try {
            while (numberOfFinishedThreads < numberOfRunningThreads) {
                wait();
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    private synchronized void finishWorkingThread(long cpuTime, long userTime) {
        this.accumulatedCpuTime += cpuTime;
        this.accumulatedUserTime += userTime;
        numberOfFinishedThreads++;
        notifyAll();
    }

    private class Task {

        private final String taskName;
        private final TaskKey taskKey;
        private TaskResult taskResult;

        private Task(String taskName, TaskKey taskKey) {
            this.taskName = taskName;
            this.taskKey = taskKey;
        }
    }

    class MultiTaskWorkingThread extends EThread {

        private MultiTaskWorkingThread(int id) {
            super("WorkingThread-" + id);
            userInterface = new ServerJobManager.UserInterfaceRedirect(ownerThread.ejob.jobKey, null);
            ejob = ownerThread.ejob;
            isServerThread = ownerThread.isServerThread;
            database = ownerThread.database;
        }

        @Override
        public void run() {
            long accumulatedTime = 0;
            Environment.setThreadEnvironment(env);
            for (;;) {
                Task t = getTask();
                if (t == null) {
                    break;
                }
                ;
                try {
                    long startTime = System.currentTimeMillis();
                    t.taskResult = runTask(t.taskKey);
                    long endTime = System.currentTimeMillis();
                    accumulatedTime += (endTime - startTime);
                } catch (Throwable e) {
                    e.getStackTrace();
                    e.printStackTrace(System.out);
                    e.printStackTrace();
                }
            }
            long cpuTime = threadMX.getCurrentThreadCpuTime();
            long userTime = threadMX.getCurrentThreadUserTime();
            System.out.println(getName() + " clock=" + accumulatedTime/MILLIS_IN_SEC + " cpu=" + cpuTime/NANOS_IN_SEC + " user=" + userTime/NANOS_IN_SEC);
            finishWorkingThread(cpuTime, userTime);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy