⛏️ index : haiku.git

/*
 * Copyright 2001-2007, Haiku.
 * Distributed under the terms of the MIT License.
 *
 * Authors:
 *		Tyler Dauwalder
 */


#include "RegistrarThreadManager.h"

#include <RegistrarDefs.h>

#include <stdio.h>

#include "RegistrarThread.h"


//#define DBG(x) x
#define DBG(x)
#define OUT printf

using namespace BPrivate;

/*!
	\class RegistrarThreadManager
	\brief RegistrarThreadManager is the master of all threads spawned by the
		registrar
*/

//! Creates a new RegistrarThreadManager object
RegistrarThreadManager::RegistrarThreadManager()
	: fThreadCount(0)
{
}

// destructor
/*! \brief Destroys the RegistrarThreadManager object, killing and deleting any
	still running threads.
*/
RegistrarThreadManager::~RegistrarThreadManager()
{
	KillThreads();
}

// MessageReceived
/*! \brief Handles \c B_REG_MIME_UPDATE_THREAD_FINISHED messages, passing on all
	others.

	Each \c B_REG_MIME_UPDATE_THREAD_FINISHED message triggers a call to
	CleanupThreads().
*/
void
RegistrarThreadManager::MessageReceived(BMessage* message)
{
	switch (message->what) {
		case B_REG_MIME_UPDATE_THREAD_FINISHED:
		{
			CleanupThreads();
			break;
		}

		default:
		{
			BHandler::MessageReceived(message);
			break;
		}
	}
}

// LaunchThread
/*! \brief Launches the given thread, passing responsiblity for it onto the
	RegistrarThreadManager object.

	\param thread Pointer to a newly allocated \c RegistrarThread object.

	If the result of the function is \c B_OK, the \c RegistrarThreadManager
	object assumes ownership of \a thread; if the result is an error code, it
	does not.

	\return
	- \c B_OK: success
	- \c B_NO_MORE_THREADS: the number of concurrently allowed threads (defined
							by RegistrarThreadManager::kThreadLimit) has
							already been reached
	- \c B_BAD_THREAD_STATE: the thread has already been launched
	- other error code: failure
*/
status_t
RegistrarThreadManager::LaunchThread(RegistrarThread *thread)
{
	status_t err = thread ? B_OK : B_BAD_VALUE;
	if (!err) {
		if (atomic_add(&fThreadCount, 1) >= kThreadLimit) {
			err = B_NO_MORE_THREADS;
			atomic_add(&fThreadCount, -1);
		}
	}

	if (!err) {
		fThreads.push_back(thread);
		err = thread->Run();
		if (err) {
			std::list<RegistrarThread*>::iterator i;
			for (i = fThreads.begin(); i != fThreads.end();) {
				if ((*i) == thread) {
					i = fThreads.erase(i);
					atomic_add(&fThreadCount, -1);
					break;
				} else
					++i;
			}
		}
	}
	if (!err)
		DBG(OUT("RegistrarThreadManager::LaunchThread(): launched new '%s'"
			" thread, id %ld\n", thread->Name(), thread->Id()));
	return err;
}

// CleanupThreads
/*! \brief Frees the resources of any threads that are no longer running

	\todo This function should perhaps be triggered periodically by a
	BMessageRunner once we have our own BMessageRunner implementation.
*/
status_t
RegistrarThreadManager::CleanupThreads()
{
	std::list<RegistrarThread*>::iterator i;
	for (i = fThreads.begin(); i != fThreads.end(); ) {
		if (*i) {
			if ((*i)->IsFinished()) {
				DBG(OUT("RegistrarThreadManager::CleanupThreads(): Cleaning up"
					" thread %ld\n", (*i)->Id()));
				RemoveThread(i);
					// adjusts i
			} else
				++i;
		} else {
			OUT("WARNING: RegistrarThreadManager::CleanupThreads(): NULL"
				" mime_update_thread_shared_data pointer found in and removed"
				" from RegistrarThreadManager::fThreads list\n");
			i = fThreads.erase(i);
		}
	}
	return B_OK;
}

// ShutdownThreads
/*! \brief Requests that any running threads quit and frees the resources
	of any threads that are no longer running.

	\todo This function should be called during system shutdown around
	the same time as global B_QUIT_REQUESTED messages are sent out.
*/
status_t
RegistrarThreadManager::ShutdownThreads()
{
	std::list<RegistrarThread*>::iterator i;
	for (i = fThreads.begin(); i != fThreads.end(); ) {
		if (*i) {
			if ((*i)->IsFinished()) {
				DBG(OUT("RegistrarThreadManager::ShutdownThreads(): Cleaning up"
					" thread %ld\n", (*i)->Id()));
				RemoveThread(i);
					// adjusts i
			} else {
				DBG(OUT("RegistrarThreadManager::ShutdownThreads(): Shutting"
					" down thread %ld\n", (*i)->Id()));
				(*i)->AskToExit();
				++i;
			}
		} else {
			OUT("WARNING: RegistrarThreadManager::ShutdownThreads(): NULL"
				" mime_update_thread_shared_data pointer found in and removed"
				" from RegistrarThreadManager::fThreads list\n");
			i = fThreads.erase(i);
		}
	}

	/*! \todo We may want to iterate back through the list at this point,
		snooze for a moment if find an unfinished thread, and kill it if
		it still isn't finished by the time we're done snoozing.
	*/

	return B_OK;
}

// KillThreads
/*! \brief Kills any running threads and frees their resources.

	\todo This function should probably be called during system shutdown
	just before kernel shutdown begins.
*/
status_t
RegistrarThreadManager::KillThreads()
{
	std::list<RegistrarThread*>::iterator i;
	for (i = fThreads.begin(); i != fThreads.end(); ) {
		if (*i) {
			if (!(*i)->IsFinished()) {
				DBG(OUT("RegistrarThreadManager::KillThreads(): Killing thread"
					" %ld\n", (*i)->Id()));
				status_t err = kill_thread((*i)->Id());
				if (err)
					OUT("WARNING: RegistrarThreadManager::KillThreads():"
						" kill_thread(%" B_PRId32 ") failed with error code"
						" 0x%" B_PRIx32 "\n", (*i)->Id(), err);
			}
			DBG(OUT("RegistrarThreadManager::KillThreads(): Cleaning up thread"
				" %ld\n", (*i)->Id()));
			RemoveThread(i);
				// adjusts i
		} else {
			OUT("WARNING: RegistrarThreadManager::KillThreads(): NULL"
				" mime_update_thread_shared_data pointer found in and removed"
				" from RegistrarThreadManager::fThreads list\n");
			i = fThreads.erase(i);
		}
	}
	return B_OK;
}

// ThreadCount
/*! \brief Returns the number of threads currently under management

	This is not necessarily a count of how many threads are actually running,
	as threads may remain in the thread list that are finished and waiting
	to be cleaned up.

	\return The number of threads currently under management
*/
uint
RegistrarThreadManager::ThreadCount() const
{
	return fThreadCount;
}

// RemoveThread
/*! \brief Deletes the given thread and removes it from the thread list.
*/
std::list<RegistrarThread*>::iterator&
RegistrarThreadManager::RemoveThread(std::list<RegistrarThread*>::iterator &i)
{
	status_t dummy;
	wait_for_thread((*i)->Id(), &dummy);

	delete *i;
	atomic_add(&fThreadCount, -1);
	return (i = fThreads.erase(i));
}