diff --git a/src/core/config/CMakeLists.txt b/src/core/config/CMakeLists.txt
index eb981de3887e662bb254de0f120b9c7f0440095f..87251d6d0e6431a3cc3cbaef883267cb00712bbc 100644
--- a/src/core/config/CMakeLists.txt
+++ b/src/core/config/CMakeLists.txt
@@ -19,6 +19,7 @@ OPTION(CONFIG_SUPPRESS_INTERRUPTS       "Target backend: Suppress interrupts" OF
 OPTION(CONFIG_FIRE_INTERRUPTS           "Target backend: Fire interrupts" OFF)
 OPTION(CONFIG_DISABLE_KEYB_INTERRUPTS   "Target backend: Suppress keyboard interrupts" OFF)
 OPTION(SERVER_PERFORMANCE_MEASURE       "Performance measurement in job-server" OFF)
+OPTION(CONFIG_FAST_BREAKPOINTS          "Enable fast breakpoints (requires breakpoint events to be enabled)" OFF)
 SET(SERVER_COMM_HOSTNAME        "localhost"  CACHE STRING "Job-server hostname or IP")
 SET(SERVER_COMM_TCP_PORT        "1111"       CACHE STRING "Job-server TCP port")
 SET(SERVER_PERF_LOG_PATH        "perf.log"   CACHE STRING "A file name for storing the server's performance log (CSV)")
diff --git a/src/core/config/FailConfig.hpp.in b/src/core/config/FailConfig.hpp.in
index 19db6e93f68c54e9eb3b9374f8b69cdd74abb243..389db854c5b1d3754c07e3a80d5789281061e8aa 100644
--- a/src/core/config/FailConfig.hpp.in
+++ b/src/core/config/FailConfig.hpp.in
@@ -15,6 +15,9 @@
 #cmakedefine CONFIG_EVENT_TRAP
 #cmakedefine CONFIG_EVENT_JUMP
 
+// Performance options
+#cmakedefine CONFIG_FAST_BREAKPOINTS
+
 // Save/restore functionality
 #cmakedefine CONFIG_SR_RESTORE
 #cmakedefine CONFIG_SR_SAVE
diff --git a/src/core/sal/BufferCache.cc b/src/core/sal/BufferCache.cc
deleted file mode 100644
index b732fb0fc17d69387c99751c06697feeda486349..0000000000000000000000000000000000000000
--- a/src/core/sal/BufferCache.cc
+++ /dev/null
@@ -1,32 +0,0 @@
-#include <algorithm>
-#include <vector>
-#include "BufferCache.hpp"
-#include "Listener.hpp"
-#include "ListenerManager.hpp"
-
-namespace fail {
-
-template<class T>
-typename BufferCache<T>::iterator BufferCache<T>::makeActive(ListenerManager &ev_list, BufferCache<T>::iterator idx)
-{
-	assert(idx != end() && "FATAL ERROR: Index larger than cache!");
-	T ev = *idx;
-	assert(ev && "FATAL ERROR: Object pointer cannot be NULL!");
-	ev->decreaseCounter();
-	if (ev->getCounter() > 0) {
-		return ++idx;
-	}
-	ev->resetCounter();
-	// Note: This is the one and only situation in which remove() should NOT
-	//       store the removed item in the delete-list.
-	ListenerManager::iterator it = std::find(ev_list.begin(), ev_list.end(), static_cast<BaseListener*>(ev));
-	ev_list.m_remove(it, true); // remove listener from buffer-list
-	ev_list.m_FireList.push_back(ev);
-	return erase(idx);
-}
-
-// Declare here whatever instances of the template you are going to use:
-template class BufferCache<BPListener*>;
-template class BufferCache<IOPortListener*>;
-
-} // end-of-namespace: fail
diff --git a/src/core/sal/BufferCache.hpp b/src/core/sal/BufferCache.hpp
deleted file mode 100644
index 7f8f814ceb13592815b4314fc98d2a8882f83603..0000000000000000000000000000000000000000
--- a/src/core/sal/BufferCache.hpp
+++ /dev/null
@@ -1,94 +0,0 @@
-#ifndef __BUFFER_CACHE_HPP__
-  #define __BUFFER_CACHE_HPP__
-
-#include <stdlib.h>
-#include <list>
-
-namespace fail {
-
-class ListenerManager;
-
-/**
- * \class BufferCache
- *
- * \brief A simple dynamic array
- *
- * This class is intended to serve as a kind of cache for the
- * untyped and therefore quite slow listener handling mechanism of Fail*.
- */
-template<class T>
-class BufferCache {
-public:
-	/**
-	 * The list type inherent to this class. Like bufferlist_t in ListenerManager.hpp,
-	 * but dynamically typed.
-	 */
-	typedef std::list<T> cachelist_t;
-	/**
-	 * The iterator of this class used to loop through the list of
-	 * added listeners. To retrieve an iterator to the first element, call
-	 * begin(). end() returns the iterator, pointing after the last element.
-	 * (This behaviour equals the STL iterator in C++.)
-	 */
-	typedef typename cachelist_t::iterator iterator;
-private:
-	cachelist_t m_Buffer; //!< The list holding the cached elements
-public:
-	BufferCache() { }
-	~BufferCache() { }
-	/**
-	 * Add an element to the array. The object pointed to remains untouched.
-	 * @param val the element to add
-	 */
-	inline void add(const T &val) { m_Buffer.push_back(val); }
-	/**
-	 * Remove an element from the array. The object pointed to remains untouched.
-	 * @param val the element to remove
-	 */
-	inline void remove(const T &val) { m_Buffer.remove(val); }
-	/**
-	 * Remove an element at a specific position. The object pointed to remains untouched.
-	 * @param val the element to remove
-	 * @return a pointer to the given element's successor if successful, a negative value otherwise
-	 */
-	inline iterator erase(iterator i) { return m_Buffer.erase(i); }
-	/**
-	 * Clears the array, removing all elements. The objects pointed to remain untouched.
-	 */
-	inline void clear() { m_Buffer.clear(); }
-	/**
-	 * Returns an iterator to the beginning of the internal data structure.
-	 * Don't forget to update the returned iterator when calling one of the
-	 * modifying methods like makeActive() or remove(). Therefore you need
-	 * to call the iterator-based variants of makeActive() and remove().
-	 * \code
-	 * [X|1|2| ... |n]
-	 *  ^
-	 * \endcode
-	 */
-	inline iterator begin() { return m_Buffer.begin(); }
-	/**
-	 * Returns an iterator to the end of the interal data structure.
-	 * Don't forget to update the returned iterator when calling one of the
-	 * modifying methods like makeActive() or remove(). Therefore you need
-	 * to call the iterator-based variants of makeActive() and remove().
-	 * \code
-	 * [1|2| ... |n]X
-	 *              ^
-	 * \endcode
-	 */
-	inline iterator end() { return m_Buffer.end(); }
-	/**
-	 * Acts as a replacement for ListenerManager::makeActive, manipulating
-	 * the buffer cache exclusively. ListenerManager::triggerActiveListeners needs
-	 * to be called to fire the active listeners (see there).
-	 * This method is declared as a friend method in ListenerManager.
-	 * @param idx the index of the listener to trigger
-	 * @returns an updated index which can be used to update a loop counter
-	 */
-	iterator makeActive(ListenerManager &ev_list, iterator idx);
-};
-
-} // end-of-namespace: fail
-
-#endif // __BUFFER_CACHE_HPP__
diff --git a/src/core/sal/CMakeLists.txt b/src/core/sal/CMakeLists.txt
index 576e54d4e2532b44d3c2c6bf7377a3494b3ae5e7..84f5a1e9b5dfe5331c8c19682903756fdd4dd530 100644
--- a/src/core/sal/CMakeLists.txt
+++ b/src/core/sal/CMakeLists.txt
@@ -1,6 +1,5 @@
 if(BUILD_BOCHS)
 	set(SRCS
-		BufferCache.cc
 		Listener.cc
 		ListenerManager.cc
 		SALConfig.cc
@@ -12,7 +11,6 @@ if(BUILD_BOCHS)
 	)
 elseif(BUILD_GEM5)
 	set(SRCS
-		BufferCache.cc
 		Listener.cc
 		ListenerManager.cc
 		SALConfig.cc
@@ -24,7 +22,6 @@ elseif(BUILD_GEM5)
 	)
 elseif(BUILD_OVP)
 	set(SRCS
-		BufferCache.cc
 		Listener.cc
 		ListenerManager.cc
 		SALConfig.cc
@@ -35,7 +32,6 @@ elseif(BUILD_OVP)
 	)
 elseif(BUILD_QEMU)
 	set(SRCS
-		BufferCache.cc
 		Listener.cc
 		ListenerManager.cc
 		SALConfig.cc
diff --git a/src/core/sal/ListenerManager.cc b/src/core/sal/ListenerManager.cc
index 265a4699bb875637c642f014a350385188d3a5fb..7bc8a7f2dc99c1e556e278eee62c990417afa1d7 100644
--- a/src/core/sal/ListenerManager.cc
+++ b/src/core/sal/ListenerManager.cc
@@ -5,34 +5,6 @@
 
 namespace fail {
 
-void ListenerManager::addToCaches(BaseListener *li) // FIXME: REMOVE
-{
-	BPListener *bps_li;
-	if ((bps_li = dynamic_cast<BPListener*>(li)) != NULL)
-		m_Bp_cache.add(bps_li);
-
-	IOPortListener *io_li;
-	if ((io_li = dynamic_cast<IOPortListener*>(li)) != NULL)
-		m_Io_cache.add(io_li);
-}
-
-void ListenerManager::removeFromCaches(BaseListener *li) // FIXME: REMOVE
-{
-	BPListener *bpr_li;
-	if ((bpr_li = dynamic_cast<BPListener*>(li)) != NULL)
-		m_Bp_cache.remove(bpr_li);
-
-	IOPortListener *io_li;
-	if ((io_li = dynamic_cast<IOPortListener*>(li)) != NULL)
-		m_Io_cache.remove(io_li);
-}
-
-void ListenerManager::clearCaches() // FIXME: REMOVE
-{
-	m_Bp_cache.clear();
-	m_Io_cache.clear();
-}
-
 void ListenerManager::add(BaseListener* li, ExperimentFlow* pExp)
 {
 	assert(li != NULL && "FATAL ERROR: Listener (of base type BaseListener*) cannot be NULL!");
@@ -47,7 +19,6 @@ void ListenerManager::add(BaseListener* li, ExperimentFlow* pExp)
 	index_t idx = m_BufferList.size()-1;
 	assert(m_BufferList[idx] == li && "FATAL ERROR: Invalid index after push_back() unexpected!");
 	li->setLocation(idx);
-	addToCaches(li); // FIXME: REMOVE
 }
 
 void ListenerManager::remove(BaseListener* li)
@@ -81,7 +52,6 @@ void ListenerManager::remove(BaseListener* li)
 				//  ==> no further processing required here
 			}
 		}
-		clearCaches(); // FIXME: REMOVE
 		// All remaining active listeners must not fire anymore
 		m_DeleteList.insert(m_DeleteList.end(), m_FireList.begin(), m_FireList.end());
 
@@ -92,7 +62,6 @@ void ListenerManager::remove(BaseListener* li)
 	//   * If 'li' in 'm_FireList', copy to 'm_DeleteList'
 	} else if (li->getLocation() != INVALID_INDEX) { // Check if li hasn't been added previously (Q&D)
 		li->onDeletion();
-		removeFromCaches(li); // FIXME: REMOVE
 		if (li->getPerformanceBuffer() != NULL)
 			li->getPerformanceBuffer()->remove(li->getLocation());
 		m_remove(li->getLocation());
@@ -118,8 +87,6 @@ void ListenerManager::m_remove(index_t idx)
 	// Note: This operation has O(1) time complexity. It copies (aka "swaps") the
 	//       trailing element "m_BufferList[m_BufferList.size()-1]" to the slot
 	//       at "m_BufferList[idx]" and removes the last element (pop_back()).
-	if(idx == INVALID_INDEX)
-		((BaseListener*)NULL)->setPerformanceBuffer(NULL);
 
 	// Override the element to be deleted (= copy the last element to the slot
 	// of the element to be deleted) and update their attributes:
@@ -156,14 +123,6 @@ ListenerManager::iterator ListenerManager::m_remove(iterator it, bool skip_delet
 		// to *delete* (not "move") a listener.
 		(*it)->onDeletion();
 		m_DeleteList.push_back(*it);
-
-		// Cached listeners have their own BufferCache<T>::makeActive() implementation, which
-		// calls this method and afterwards erase() in the cache class. This is why, when
-		// called from any kind of makeActive() method, it is unnecessary to call
-		// BufferCache<T>::remove() from m_remove().
-
-		// NOTE: in case the semantics of skip_deletelist change, please adapt the following lines
-		removeFromCaches(*it); // FIXME: REMOVE (incl. comments)
 	}
 
 	// This has O(1) time complexity due to a underlying std::vector (-> random access iterator)
@@ -206,7 +165,6 @@ void ListenerManager::remove(ExperimentFlow* flow)
 		for (std::set<PerfBufferBase*>::iterator it = perfBufLists.begin();
 			 it != perfBufLists.end(); ++it)
 			(*it)->clear();
-		clearCaches(); // FIXME: REMOVE
 		// All remaining active listeners must not fire anymore
 		m_DeleteList.insert(m_DeleteList.end(), m_FireList.begin(), m_FireList.end());
 	} else { // remove all listeners corresponding to a specific experiment ("flow"):
diff --git a/src/core/sal/ListenerManager.hpp b/src/core/sal/ListenerManager.hpp
index 2201e26b6bfad556128d3a31b0528c20857e2461..a7f2768bd9839f3ac521457147cca2b0f8917328 100644
--- a/src/core/sal/ListenerManager.hpp
+++ b/src/core/sal/ListenerManager.hpp
@@ -7,7 +7,6 @@
 #include <algorithm>
 
 #include "Listener.hpp"
-#include "BufferCache.hpp"
 
 namespace fail {
 
@@ -30,14 +29,6 @@ typedef std::vector<BaseListener*>  firelist_t;
  */
 typedef std::vector<BaseListener*>  deletelist_t;
 
-/**
- * Cache classes for the most commonly used types of listeners, utilising static typing.
- * Apart from that, they work like bufferlist_t.
- */
-typedef BufferCache<BPListener*> bp_cache_t;
-typedef bp_cache_t::iterator bp_iter_t;
-typedef BufferCache<IOPortListener*> io_cache_t;
-typedef io_cache_t::iterator io_iter_t;
 /**
  * \class ListenerManager
  *
@@ -59,10 +50,6 @@ private:
 	firelist_t m_FireList; //!< the active listeners (used temporarily)
 	deletelist_t m_DeleteList; //!< the deleted listeners (used temporarily)
 	BaseListener* m_pFired; //!< the recently fired Listener-object
-	bp_cache_t m_Bp_cache; //!< the storage cache for breakpoint listeners
-	io_cache_t m_Io_cache; //!< the storage cache for port i/o listeners
-	friend bp_iter_t bp_cache_t::makeActive(ListenerManager &ev_list, bp_iter_t idx);
-	friend io_iter_t io_cache_t::makeActive(ListenerManager &ev_list, io_iter_t idx);
 public:
 	/**
 	 * Determines the pointer to the listener base type, stored at index \c idx.
@@ -124,7 +111,7 @@ private:
 	 * Updates the buffer-list by "removing" the element located at index \c idx.
 	 * This is done by replacing the element with the last element of the vector.
 	 * @param idx the index of the element to be removed
-	 * @warning The internals of the listener, stored at index \c idx wont be
+	 * @warning The internals of the listener, stored at index \c idx will be
 	 *          updated.
 	 * @note This method should typically be used in a performance buffer-list
 	 *       implemenation.
@@ -226,31 +213,6 @@ public:
 	 * triggered, the (internal) fire- and delete-list will be cleared.
 	 */
 	void triggerActiveListeners();
-	/**
-	 * Retrieves the BPListener buffer cache.
-	 * @returns the buffer cache
-	 */
-	inline bp_cache_t &getBPBuffer() { return m_Bp_cache; }
-	/**
-	 * Retrieves the IOPortListener buffer cache.
-	 * @returns the buffer cache
-	 */
-	inline io_cache_t &getIOBuffer() { return m_Io_cache; }
-private:
-	/**
-	 * Add an listener to its appropriate cache.
-	 * @param li the listener to add
-	 */
-	void addToCaches(BaseListener* li);
-	/**
-	 * Remove an listener from its cache.
-	 * @param li the listener to remove
-	 */
-	void removeFromCaches(BaseListener* li);
-	/**
-	 * Clear the listener caches.
-	 */
-	void clearCaches();
 };
 
 } // end-of-namespace: fail
diff --git a/src/core/sal/bochs/BochsController.cc b/src/core/sal/bochs/BochsController.cc
index 9d9bb9f92a5b7378bf4ba7baa2bef90e0c3b9423..a796cae2c92d6f8936107fbb8d409f1aa67fdf8f 100644
--- a/src/core/sal/bochs/BochsController.cc
+++ b/src/core/sal/bochs/BochsController.cc
@@ -95,14 +95,14 @@ void BochsController::onBreakpoint(address_t instrPtr, address_t address_space)
 #endif
 	bool do_fire = false;
 	// Check for active breakpoint-events:
-	bp_cache_t &buffer_cache = m_LstList.getBPBuffer();
-	bp_cache_t::iterator it = buffer_cache.begin();
+	ListenerManager::iterator it = m_LstList.begin();
 	BPEvent tmp(instrPtr, address_space);
-	while (it != buffer_cache.end()) {
-		BPListener* pEvBreakpt = *it;
-		if (pEvBreakpt->isMatching(&tmp)) {
-			pEvBreakpt->setTriggerInstructionPointer(instrPtr);
-			it = buffer_cache.makeActive(m_LstList, it);
+	while (it != m_LstList.end()) {
+		BaseListener* pLi = *it;
+		BPListener* pBreakpt = dynamic_cast<BPListener*>(pLi);
+		if (pBreakpt != NULL && pBreakpt->isMatching(&tmp)) {
+			pBreakpt->setTriggerInstructionPointer(instrPtr);
+			it = m_LstList.makeActive(it);
 			do_fire = true;
 			// "it" has already been set to the next element (by calling
 			// makeActive()):
@@ -126,13 +126,13 @@ void BochsController::updateBPEventInfo(BX_CPU_C *context, bxICacheEntry_c *cach
 
 void BochsController::onIOPort(unsigned char data, unsigned port, bool out) {
 	// Check for active IOPortListeners:
-	io_cache_t &buffer_cache = m_LstList.getIOBuffer();
-	io_cache_t::iterator it = buffer_cache.begin();
-	while (it != buffer_cache.end()) {
-		IOPortListener* pIOPt = (*it);
-		if (pIOPt->isMatching(port, out)) {
+	ListenerManager::iterator it = m_LstList.begin();
+	while (it != m_LstList.end()) {
+		BaseListener* pLi = *it;
+		IOPortListener* pIOPt = dynamic_cast<IOPortListener*>(pLi);
+		if (pIOPt != NULL && pIOPt->isMatching(port, out)) {
 			pIOPt->setData(data);
-			it = buffer_cache.makeActive(m_LstList, it);
+			it = m_LstList.makeActive(it);
 			// "it" has already been set to the next element (by calling
 			// makeActive()):
 			continue; // -> skip iterator increment
diff --git a/src/core/sal/gem5/Gem5Controller.cc b/src/core/sal/gem5/Gem5Controller.cc
index d5490d421f2df7950b735d31aecfbee70a0a1cf1..9535df5ace9bfaa42394ae7445b3a314df517a3e 100644
--- a/src/core/sal/gem5/Gem5Controller.cc
+++ b/src/core/sal/gem5/Gem5Controller.cc
@@ -35,17 +35,16 @@ void Gem5Controller::reboot()
 void Gem5Controller::onBreakpoint(address_t instrPtr, address_t address_space)
 {
 	bool do_fire = false;
+	BPEvent tmp(instrPtr, address_space);
 	// Check for active breakpoint-events:
-	bp_cache_t &buffer_cache = m_LstList.getBPBuffer();
-	for(bp_cache_t::iterator it = buffer_cache.begin(); it != buffer_cache.end(); it++)
-	{
-		BPListener* pEvBreakpt = *it;
-		if(pEvBreakpt->isMatching(instrPtr, address_space)) {
-			pEvBreakpt->setTriggerInstructionPointer(instrPtr);
-			it = buffer_cache.makeActive(m_LstList, it);
+	for (ListenerManager::iterator it = m_LstList.begin(); it != m_LstList.end(); it++) {
+		BaseListener* pLi = *it;
+		BPListener* pBreakpt = dynamic_cast<BPListener*>(pLi);
+		if(pBreakpt != NULL && pBreakpt->isMatching(&tmp)) {
+			pBreakpt->setTriggerInstructionPointer(instrPtr);
+			it = m_LstList.makeActive(it);
 			do_fire = true;
-			// "it" has already been set to the next element (by calling
-			// makeActive()):
+			// "it" has already been set to the next element (by calling makeActive()):
 			continue; // -> skip iterator increment
 		}
 	}
diff --git a/src/core/sal/qemu/QEMUController.cc b/src/core/sal/qemu/QEMUController.cc
index c6345dab1e42cbd3381cb60ed175806bf82657ae..1c68e06c3c386fe32119bfa3ece23b83605837cd 100644
--- a/src/core/sal/qemu/QEMUController.cc
+++ b/src/core/sal/qemu/QEMUController.cc
@@ -22,15 +22,16 @@ QEMUController::~QEMUController()
 }
 
 // FIXME: copied from BochsController; remove redundancy!
-void QEMUController::onIOPort(unsigned char data, unsigned port, bool out) {
+void QEMUController::onIOPort(unsigned char data, unsigned port, bool out)
+{
 	// Check for active IOPortListeners:
-	io_cache_t &buffer_cache = m_LstList.getIOBuffer();
-	io_cache_t::iterator it = buffer_cache.begin();
-	while (it != buffer_cache.end()) {
-		IOPortListener* pIOPt = (*it);
-		if (pIOPt->isMatching(port, out)) {
+	ListenerManager::iterator it = m_LstList.begin();
+	while (it != m_LstList.end()) {
+		BaseListener* pLi = ;
+		IOPortListener* pIOPt = dynamic_cast<IOPortListener>(*it);
+		if (pIOPt != NULL && pIOPt->isMatching(port, out)) {
 			pIOPt->setData(data);
-			it = buffer_cache.makeActive(m_LstList, it);
+			it = m_LstList.makeActive(it);
 			// "it" has already been set to the next element (by calling
 			// makeActive()):
 			continue; // -> skip iterator increment