Work-in-progress of the module notification implementation:

* Most things are working, it just doesn't notify the listeners on changes yet.


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@26342 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Axel Dörfler 2008-07-09 14:23:29 +00:00
parent bb73c05fd7
commit daa861bc29

View File

@ -19,15 +19,20 @@
#include <FindDirectory.h>
#include <NodeMonitor.h>
#include <dirent_private.h>
#include <boot_device.h>
#include <boot/elf.h>
#include <elf.h>
#include <fs/KPath.h>
#include <fs/node_monitor.h>
#include <lock.h>
#include <Notifications.h>
#include <safemode.h>
#include <syscalls.h>
#include <util/AutoLock.h>
#include <util/khash.h>
#include <util/Stack.h>
#include <vfs.h>
@ -138,34 +143,50 @@ typedef struct module_iterator {
namespace Module {
struct module_listener : DoublyLinkedListLink<module_listener> {
struct entry {
dev_t device;
ino_t node;
};
struct hash_entry : entry {
HashTableLink<hash_entry> link;
};
struct NodeHashDefinition {
typedef entry* KeyType;
typedef hash_entry ValueType;
size_t Hash(ValueType* entry) const
{ return HashKey(entry); }
HashTableLink<ValueType>* GetLink(ValueType* entry) const
{ return &entry->link; }
size_t HashKey(KeyType key) const
{
return ((uint32)(key->node >> 32) + (uint32)key->node) ^ key->device;
}
bool Compare(KeyType key, ValueType* entry) const
{
return key->device == entry->device
&& key->node == entry->node;
}
};
typedef OpenHashTable<NodeHashDefinition> NodeHash;
struct module_listener : DoublyLinkedListLinkImpl<module_listener> {
~module_listener()
{
free((char*)prefix);
}
NotificationListener* listener;
const char* prefix;
};
typedef DoublyLinkedList<module_listener> ModuleListenerList;
class ModuleNotificationService : public NotificationService {
public:
ModuleNotificationService();
virtual ~ModuleNotificationService();
status_t InitCheck();
status_t AddListener(const KMessage* eventSpecifier,
NotificationListener& listener);
status_t UpdateListener(const KMessage* eventSpecifier,
NotificationListener& listener);
status_t RemoveListener(const KMessage* eventSpecifier,
NotificationListener& listener);
virtual const char* Name() { return "modules"; }
private:
recursive_lock fLock;
ModuleListenerList fListeners;
};
class DirectoryWatcher : public NotificationListener {
public:
DirectoryWatcher();
@ -184,6 +205,43 @@ public:
const KMessage* event);
};
class ModuleNotificationService : public NotificationService {
public:
ModuleNotificationService();
virtual ~ModuleNotificationService();
status_t InitCheck();
status_t AddListener(const KMessage* eventSpecifier,
NotificationListener& listener);
status_t UpdateListener(const KMessage* eventSpecifier,
NotificationListener& listener);
status_t RemoveListener(const KMessage* eventSpecifier,
NotificationListener& listener);
bool HasNode(dev_t device, ino_t node);
virtual const char* Name() { return "modules"; }
private:
status_t _AddNode(dev_t device, ino_t node, uint32 flags,
NotificationListener& listener);
status_t _AddDirectoryNode(dev_t device, ino_t node);
status_t _AddModuleNode(dev_t device, ino_t node);
status_t _AddDirectory(const char* prefix);
status_t _ScanDirectory(char* directoryPath, const char* prefix,
size_t& prefixPosition);
status_t _ScanDirectory(Stack<DIR*>& stack, DIR* dir,
const char* prefix, size_t prefixPosition);
recursive_lock fLock;
ModuleListenerList fListeners;
NodeHash fNodes;
DirectoryWatcher fDirectoryWatcher;
ModuleWatcher fModuleWatcher;
};
} // namespace Module
using namespace Module;
@ -926,7 +984,7 @@ nextModuleImage:
while (*iterator->current_header != NULL) {
module_info* info = *iterator->current_header;
// TODO: we mightS want to create a module here and cache it in the
// TODO: we might want to create a module here and cache it in the
// hash table
iterator->current_header++;
@ -1124,8 +1182,14 @@ DirectoryWatcher::EventOccured(NotificationService& service,
if (opcode == B_ENTRY_MOVED) {
// Determine wether it's a move within, out of, or into one
// of our watched directories.
ino_t from = event->GetInt64("from directory", -1);
ino_t to = event->GetInt64("to directory", -1);
directory = event->GetInt64("to directory", -1);
if (!sModuleNotificationService.HasNode(device, directory)) {
directory = event->GetInt64("from directory", -1);
opcode = B_ENTRY_REMOVED;
} else {
// Move within, doesn't sound like a good idea for modules
opcode = B_ENTRY_CREATED;
}
}
KPath path(B_PATH_NAME_LENGTH + 1);
@ -1137,6 +1201,7 @@ DirectoryWatcher::EventOccured(NotificationService& service,
dprintf("module \"%s\" %s\n", path.Leaf(),
opcode == B_ENTRY_CREATED ? "added" : "removed");
//sModuleNotificationService.Notify(path.Path(), opcode);
}
@ -1181,7 +1246,30 @@ status_t
ModuleNotificationService::AddListener(const KMessage* eventSpecifier,
NotificationListener& listener)
{
return B_ERROR;
const char* prefix = eventSpecifier->GetString("prefix", NULL);
if (prefix == NULL)
return B_BAD_VALUE;
module_listener* moduleListener = new(std::nothrow) module_listener;
if (moduleListener == NULL)
return B_NO_MEMORY;
moduleListener->prefix = strdup(prefix);
if (moduleListener->prefix == NULL) {
delete moduleListener;
return B_NO_MEMORY;
}
status_t status = _AddDirectory(prefix);
if (status != B_OK) {
delete moduleListener;
return status;
}
moduleListener->listener = &listener;
fListeners.Add(moduleListener);
return B_OK;
}
@ -1201,6 +1289,205 @@ ModuleNotificationService::RemoveListener(const KMessage* eventSpecifier,
}
bool
ModuleNotificationService::HasNode(dev_t device, ino_t node)
{
RecursiveLocker _(fLock);
struct entry entry = {device, node};
return fNodes.Lookup(&entry) != NULL;
}
status_t
ModuleNotificationService::_AddNode(dev_t device, ino_t node, uint32 flags,
NotificationListener& listener)
{
RecursiveLocker locker(fLock);
if (HasNode(device, node))
return B_OK;
struct hash_entry* entry = new(std::nothrow) hash_entry;
if (entry == NULL)
return B_NO_MEMORY;
status_t status = add_node_listener(device, node, flags, listener);
if (status != B_OK) {
delete entry;
return status;
}
//dprintf(" add %s %ld:%Ld\n", flags == B_WATCH_DIRECTORY ? "dir" : "file", device, node);
entry->device = device;
entry->node = node;
fNodes.Insert(entry);
return B_OK;
}
status_t
ModuleNotificationService::_AddDirectoryNode(dev_t device, ino_t node)
{
return _AddNode(device, node, B_WATCH_DIRECTORY, fDirectoryWatcher);
}
status_t
ModuleNotificationService::_AddModuleNode(dev_t device, ino_t node)
{
return _AddNode(device, node, B_WATCH_STAT, fModuleWatcher);
}
status_t
ModuleNotificationService::_AddDirectory(const char* prefix)
{
for (uint32 i = 0; i < kNumModulePaths; i++) {
if (sDisableUserAddOns && i >= kFirstNonSystemModulePath)
break;
KPath pathBuffer;
if (find_directory(kModulePaths[i], gBootDevice, true,
pathBuffer.LockBuffer(), pathBuffer.BufferSize()) != B_OK)
continue;
pathBuffer.UnlockBuffer();
pathBuffer.Append("kernel");
pathBuffer.Append(prefix);
size_t prefixPosition = strlen(prefix);
status_t status = _ScanDirectory(pathBuffer.LockBuffer(), prefix,
prefixPosition);
pathBuffer.UnlockBuffer();
if (status != B_OK)
return status;
}
return B_OK;
}
status_t
ModuleNotificationService::_ScanDirectory(char* directoryPath,
const char* prefix, size_t& prefixPosition)
{
DIR* dir = NULL;
while (true) {
dir = opendir(directoryPath);
if (dir != NULL || prefixPosition == 0)
break;
// the full prefix is not accessible, remove path components
const char* parentPrefix = prefix + prefixPosition - 1;
while (parentPrefix != prefix && parentPrefix[0] != '/')
parentPrefix--;
size_t cutPosition = parentPrefix - prefix;
size_t length = strlen(directoryPath);
directoryPath[length - prefixPosition + cutPosition] = '\0';
prefixPosition = cutPosition;
}
if (dir == NULL)
return B_ERROR;
Stack<DIR*> stack;
stack.Push(dir);
while (stack.Pop(&dir)) {
status_t status = _ScanDirectory(stack, dir, prefix, prefixPosition);
if (status != B_OK)
return status;
}
return B_OK;
}
status_t
ModuleNotificationService::_ScanDirectory(Stack<DIR*>& stack, DIR* dir,
const char* prefix, size_t prefixPosition)
{
bool directMatchAdded = false;
struct dirent* dirent;
while ((dirent = readdir(dir)) != NULL) {
if (dirent->d_name[0] == '.')
continue;
bool directMatch = false;
if (prefix[prefixPosition] != '\0') {
// the start must match
const char* startPrefix = prefix + prefixPosition;
if (startPrefix[0] == '/')
startPrefix++;
const char* endPrefix = strchr(startPrefix, '/');
size_t length;
if (endPrefix != NULL)
length = endPrefix - startPrefix;
else
length = strlen(startPrefix);
if (strncmp(dirent->d_name, startPrefix, length))
continue;
if (dirent->d_name[length] == '\0')
directMatch = true;
}
struct stat stat;
status_t status = vfs_read_stat(dir->fd, dirent->d_name, true, &stat,
true);
if (status != B_OK)
continue;
if (S_ISDIR(stat.st_mode)) {
int fd = _kern_open_dir(dir->fd, dirent->d_name);
if (fd < 0)
continue;
DIR* subDir = (DIR*)malloc(DIR_BUFFER_SIZE);
if (subDir == NULL) {
close(fd);
continue;
}
subDir->fd = fd;
subDir->entries_left = 0;
stack.Push(subDir);
if (_AddDirectoryNode(stat.st_dev, stat.st_ino) == B_OK
&& directMatch)
directMatchAdded = true;
} else if (S_ISREG(stat.st_mode)) {
if (_AddModuleNode(stat.st_dev, stat.st_ino) == B_OK && directMatch)
directMatchAdded = true;
}
}
if (!directMatchAdded) {
// We need to monitor this directory to see if a matching file
// is added.
struct stat stat;
status_t status = vfs_read_stat(dir->fd, NULL, true, &stat, true);
if (status == B_OK)
_AddDirectoryNode(stat.st_dev, stat.st_ino);
}
closedir(dir);
return B_OK;
}
// #pragma mark - Exported Kernel API (private part)