Package org.jscsi.scsi.tasks.management

Source Code of org.jscsi.scsi.tasks.management.DefaultTaskRouter

/**
* Copyright (c) 2012, University of Konstanz, Distributed Systems Group
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*     * Redistributions of source code must retain the above copyright
*       notice, this list of conditions and the following disclaimer.
*     * Redistributions in binary form must reproduce the above copyright
*       notice, this list of conditions and the following disclaimer in the
*       documentation and/or other materials provided with the distribution.
*     * Neither the name of the University of Konstanz nor the
*       names of its contributors may be used to endorse or promote products
*       derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
//Cleversafe open-source code header - Version 1.1 - December 1, 2006
//
//Cleversafe Dispersed Storage(TM) is software for secure, private and
//reliable storage of the world's data using information dispersal.
//
//Copyright (C) 2005-2007 Cleversafe, 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 2
//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, write to the Free Software
//Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301,
//USA.
//
//Contact Information:
// Cleversafe, 10 W. 35th Street, 16th Floor #84,
// Chicago IL 60616
// email: licensing@cleversafe.org
//
//END-OF-HEADER
//-----------------------
//@author: John Quigley <jquigley@cleversafe.com>
//@date: January 1, 2008
//---------------------

package org.jscsi.scsi.tasks.management;

import java.nio.ByteBuffer;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

import org.apache.log4j.Logger;
import org.jscsi.core.scsi.Status;
import org.jscsi.scsi.lu.LogicalUnit;
import org.jscsi.scsi.protocol.Command;
import org.jscsi.scsi.protocol.sense.exceptions.IllegalRequestException;
import org.jscsi.scsi.protocol.sense.exceptions.LogicalUnitNotSupportedException;
import org.jscsi.scsi.tasks.Task;
import org.jscsi.scsi.tasks.TaskFactory;
import org.jscsi.scsi.tasks.TaskRouter;
import org.jscsi.scsi.tasks.target.TargetTaskFactory;
import org.jscsi.scsi.transport.Nexus;
import org.jscsi.scsi.transport.TargetTransportPort;

public class DefaultTaskRouter implements TaskRouter
{
   private static Logger _logger = Logger.getLogger(DefaultTaskRouter.class);

   private static int DEFAULT_TARGET_THREAD_COUNT = 1;
   private static int DEFAULT_TARGET_QUEUE_LENGTH = 32;

   ////////////////////////////////////////////////////////////////////////////
   // data members

   private TaskSet targetTaskSet;
   private TaskManager targetTaskManager;
   private Map<Long, LogicalUnit> logicalUnitMap;
   private TaskFactory taskFactory;

   private Thread manager;

   private boolean running;

   ////////////////////////////////////////////////////////////////////////////
   // constructor(s)

   public DefaultTaskRouter()
   {
      this.logicalUnitMap = new ConcurrentHashMap<Long, LogicalUnit>();
      this.taskFactory = new TargetTaskFactory(this.logicalUnitMap.keySet());
      this.targetTaskSet = new DefaultTaskSet(DEFAULT_TARGET_QUEUE_LENGTH);
      this.targetTaskManager = new DefaultTaskManager(DEFAULT_TARGET_THREAD_COUNT, targetTaskSet);
   }

   public DefaultTaskRouter(int targetQueueLength, int targetThreadCount)
   {
      this.logicalUnitMap = new ConcurrentHashMap<Long, LogicalUnit>();
      this.taskFactory = new TargetTaskFactory(this.logicalUnitMap.keySet());
      this.targetTaskSet = new DefaultTaskSet(targetQueueLength);
      this.targetTaskManager = new DefaultTaskManager(targetThreadCount, targetTaskSet);
   }

   public DefaultTaskRouter(
         TaskFactory targetTaskFactory,
         TaskSet targetTaskSet,
         TaskManager targetTaskManager)
   {
      this.logicalUnitMap = new ConcurrentHashMap<Long, LogicalUnit>();
      this.taskFactory = targetTaskFactory;
      this.targetTaskManager = targetTaskManager;
      this.targetTaskSet = targetTaskSet;
   }

   ////////////////////////////////////////////////////////////////////////////
   // TaskRouter implementation

   public void enqueue(TargetTransportPort port, Command command)
   {
      long lun = command.getNexus().getLogicalUnitNumber();

      if (lun < 0)
      {
         try
         {
            Task task = this.taskFactory.getInstance(port, command);
            assert task != null : "improper task factory implementation returned null task";
            this.targetTaskSet.offer(task); // non-blocking, sends any errors to transport port
            if (_logger.isDebugEnabled())
               _logger.debug("successfully enqueued target command with TaskRouter: " + command);
         }
         catch (IllegalRequestException e)
         {
            _logger.error("error when parsing command: " + e);
            port.writeResponse(command.getNexus(), command.getCommandReferenceNumber(),
                  Status.CHECK_CONDITION, ByteBuffer.wrap(e.encode()));
         }
      }
      else if (logicalUnitMap.containsKey(lun))
      {
         logicalUnitMap.get(lun).enqueue(port, command);
         if (_logger.isDebugEnabled())
            _logger.debug("successfully enqueued command to logical unit with TaskRouter: "
                  + command);
      }
      else
      {
         port.writeResponse(command.getNexus(), command.getCommandReferenceNumber(),
               Status.CHECK_CONDITION,
               ByteBuffer.wrap((new LogicalUnitNotSupportedException()).encode()));
      }
   }

   public TaskServiceResponse execute(Nexus nexus, TaskManagementFunction function)
   {
      long lun = nexus.getLogicalUnitNumber();
      switch (function)
      {
         case ABORT_TASK :
            // Need an I_T_L_Q nexus with a valid LUN
            if (lun < 0 || nexus.getTaskTag() < 0 || !this.logicalUnitMap.containsKey(lun))
               return TaskServiceResponse.FUNCTION_REJECTED;
            else
               return this.logicalUnitMap.get(lun).abortTask(nexus);

         case ABORT_TASK_SET :
            // Need an I_T_L nexus with a valid LUN
            if (lun < 0 || !this.logicalUnitMap.containsKey(lun))
               return TaskServiceResponse.FUNCTION_REJECTED;
            else
               return this.logicalUnitMap.get(lun).abortTaskSet(nexus);

         case CLEAR_TASK_SET :
            // Need an I_T_L nexus with a valid LUN
            if (lun < 0 || !this.logicalUnitMap.containsKey(lun))
               return TaskServiceResponse.FUNCTION_REJECTED;
            else
               return this.logicalUnitMap.get(lun).clearTaskSet(nexus);

         case LOGICAL_UNIT_RESET :
            // Need an I_T_L nexus with a valid LUN
            if (lun < 0 || !this.logicalUnitMap.containsKey(lun))
               return TaskServiceResponse.FUNCTION_REJECTED;
            else
               return this.logicalUnitMap.get(lun).reset();

         case TARGET_RESET :
            // reset all logical units and abort target task set
            for (LogicalUnit lu : this.logicalUnitMap.values())
            {
               lu.reset();
            }
            this.targetTaskSet.clear();
            return TaskServiceResponse.FUNCTION_COMPLETE;

         case CLEAR_ACA :
            return TaskServiceResponse.FUNCTION_REJECTED;

         case WAKEUP :
            return TaskServiceResponse.FUNCTION_REJECTED;

         default :
            return TaskServiceResponse.FUNCTION_REJECTED;
      }
   }

   public void nexusLost()
   {
      // reset all logical units and abort target task set
      for (LogicalUnit lu : this.logicalUnitMap.values())
      {
         lu.reset();
      }
      this.targetTaskSet.clear();
   }

   public synchronized void registerLogicalUnit(long id, LogicalUnit lu)
   {
      if (this.running)
      {
         lu.start();
         _logger.debug("logical unit started: " + lu);
      }
      else
      {
         _logger.warn("not starting logical unit since router not running: " + lu);
      }

      logicalUnitMap.put(id, lu);
      if (_logger.isDebugEnabled())
         _logger.debug("registering logical unit: " + lu + " (id: " + id + ")");
   }

   public synchronized LogicalUnit removeLogicalUnit(long id)
   {
      LogicalUnit discardedLU = logicalUnitMap.remove(id);
      discardedLU.stop();
      if (_logger.isDebugEnabled())
         _logger.debug("removing logical unit: " + discardedLU);
      return discardedLU;
   }

   public synchronized void start()
   {
      if (this.running)
         return;

      for (Map.Entry<Long, LogicalUnit> lu : this.logicalUnitMap.entrySet())
      {
         _logger.debug("Starting Logical Unit " + lu.getKey());
         lu.getValue().start();
      }

      this.manager = new Thread(this.targetTaskManager, "TargetTaskManager");
      this.manager.start();

      this.running = true;
   }

   public synchronized void stop()
   {
      if (!this.running)
         return;

      for (Map.Entry<Long, LogicalUnit> lu : this.logicalUnitMap.entrySet())
      {
         _logger.debug("Stopping Logical Unit " + lu.getKey());
         lu.getValue().stop();
         _logger.debug("Logical Unit " + lu.getKey() + " finished");
      }

      _logger.debug("Signalling router task manager to stop");
      this.manager.interrupt();
      try
      {
         _logger.debug("Waiting for router task manager to terminate");
         this.manager.join();
         _logger.debug("router task manger finished");
      }
      catch (InterruptedException e)
      {
         _logger.debug("Interrupted while waiting for router task manager to finish");
      }

      this.running = false;
   }

}
TOP

Related Classes of org.jscsi.scsi.tasks.management.DefaultTaskRouter

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.