Missing files from the previous two commits. Ooops!

This commit is contained in:
Jonas 'Sortie' Termansen 2011-11-21 00:19:55 +01:00
parent 0b90ab534f
commit 313079483a
3 changed files with 206 additions and 0 deletions

View File

@ -0,0 +1,48 @@
/******************************************************************************
COPYRIGHT(C) JONAS 'SORTIE' TERMANSEN 2011.
This file is part of LibMaxsi.
LibMaxsi is free software: you can redistribute it and/or modify it under
the terms of the GNU Lesser General Public License as published by the Free
Software Foundation, either version 3 of the License, or (at your option)
any later version.
LibMaxsi 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 Lesser General Public License for
more details.
You should have received a copy of the GNU Lesser General Public License
along with LibMaxsi. If not, see <http://www.gnu.org/licenses/>.
sys/readdirents.h
Allows reading directory entries directly from a file descriptor.
******************************************************************************/
#ifndef _SYS_READDIRENTS_H
#define _SYS_READDIRENTS_H 1
#include <features.h>
__BEGIN_DECLS
@include(size_t.h)
// Keep this up to date with <sortix/directory.h>
struct sortix_dirent
{
struct sortix_dirent* d_next;
unsigned char d_type;
size_t d_namelen;
char d_name[];
};
int readdirents(int fd, struct sortix_dirent* dirent, size_t size);
__END_DECLS
#endif

94
sortix/directory.cpp Normal file
View File

@ -0,0 +1,94 @@
/******************************************************************************
COPYRIGHT(C) JONAS 'SORTIE' TERMANSEN 2011.
This file is part of Sortix.
Sortix 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.
Sortix 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 Sortix. If not, see <http://www.gnu.org/licenses/>.
filesystem.cpp
Allows access to stored sequences of bytes in an orderly fashion.
******************************************************************************/
#include "platform.h"
#include <libmaxsi/error.h>
#include <libmaxsi/memory.h>
#include <libmaxsi/string.h>
#include "syscall.h"
#include "process.h"
#include "device.h"
#include "directory.h"
using namespace Maxsi;
namespace Sortix
{
namespace Directory
{
int SysReadDirEnts(int fd, sortix_dirent* dirent, size_t size)
{
Process* process = CurrentProcess();
Device* dev = process->descriptors.Get(fd);
if ( !dev ) { Error::Set(Error::EBADF); return -1; }
if ( !dev->IsType(Device::DIRECTORY) ) { Error::Set(Error::EBADF); return -1; }
DevDirectory* dir = (DevDirectory*) dev;
sortix_dirent* prev = NULL;
while ( true )
{
// Check if we got enough bytes left to tell how many bytes we
// actually needed.
if ( size < sizeof(sortix_dirent) )
{
if ( prev ) { return 0; } // We did some work.
Error::Set(Error::EINVAL); // Nope, userspace was cheap.
return -1;
}
// Attempt to read into the space left, and if we managed to
// read at least one record, just say we succeded. The user
// space buffer is empty on the next call, so that'll probably
// succeed. The directory read function will store the number of
// bytes needed in the d_namelen variable and set errno to
// EINVAL such that userspace knows we need a larger buffer.
if ( dir->Read(dirent, size) ) { return (prev) ? 0 : -1; }
// Insert the current dirent into the single-linked list for
// easy iteration by userspace.
prev->d_next = dirent;
dirent->d_next = NULL;
// Check for end-of-directory conditions. Signal to userspace
// that we are done by giving them the empty filename.
if ( dirent->d_namelen == 0 ) { return 0; }
// Alright, we managed to read a dirent. Now let's try reading
// another one (provide as many as we can).
prev = dirent;
size_t bytesused = sizeof(sortix_dirent) + dirent->d_namelen + 1;
size -= bytesused;
dirent = (sortix_dirent*) ( ((byte*) dirent) + bytesused );
}
}
void Init()
{
Syscall::Register(SYSCALL_READDIRENTS, (void*) SysReadDirEnts);
}
}
}

64
sortix/directory.h Normal file
View File

@ -0,0 +1,64 @@
/******************************************************************************
COPYRIGHT(C) JONAS 'SORTIE' TERMANSEN 2011.
This file is part of Sortix.
Sortix 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.
Sortix 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 Sortix. If not, see <http://www.gnu.org/licenses/>.
directory.h
A container for files and other directories.
******************************************************************************/
#ifndef SORTIX_DIRECTORY_H
#define SORTIX_DIRECTORY_H
#include "device.h"
namespace Sortix
{
// Keep this up to date with <sys/readdirents.h>
struct sortix_dirent
{
struct sortix_dirent* d_next;
unsigned char d_type;
size_t d_namelen;
char d_name[];
};
class DevDirectory : public Device
{
public:
typedef Device BaseClass;
public:
virtual void Rewind() = 0;
// Precondition: available is at least sizeof(sortix_dirent).
virtual int Read(sortix_dirent* dirent, size_t available) = 0;
public:
virtual bool IsType(unsigned type) { return type == Device::DIRECTORY; }
};
namespace Directory
{
void Init();
}
}
#endif