Yeah, if you really wanted the GDT::init() style, then just use C-style global functions and variables inside of a namespace.
Code: Select all
namespace GDT
{
extern void init();
}
But I don't find this very elegant, since realistically that's the same as doing GDT_init in C except replace the '_' with '::'.
Also if you're going to use OO design, you could go with a few other of C++ features. Templates are cool, I recommend you make your own array/collection class, like std::map (or std::vector at a minimum). I like maps since they allow really fast lookup - the key could be anything from a process/thread ID, and if you don't know what to use as a key then you can use the address (you can delete something really fast without iterating through the array). Make your own std::string style class, really useful so you don't have to stick with predefined buffer sizes.
Deciding what should be classes/structs and if they should have member functions required a few large redesigns. In my first design, processes and threads were classes and you could call process->switchToContext() but I didn't really like the design after the novelty of using C++ wore off. Now I have "managers" which are classes (e.g. ProcessManager, MemoryManager, InterruptManager), and the processes, threads, etc are structs that hold only data, no member functions. I found it much more elegant to call MemoryManager->switchToProcessContext(myProcess).
You could inherit all of these managers from an interface (IManager), with common functions (initialise, count the memory used, cleanup) and loop through an array of them, but usually these managers have to be started/cleaned-up in a certain order. I don't find it worth the effort of doing this if I only have a few managers (I'm writing a microkernel, so I just have ProcessManager, MemoryManager, DriverManager, InterruptManager - perhaps a UserManager in the future, all the FS and window managing code is done in a server).
You could also have an object factory (kinda overkill for my kernel which I'm trying to keep as minimalistic as possible, but a possibility to keep control of a huge monolithic kernel). Basically, instead of typing "new Process()" you could do ObjectFactory::Singleton.CreateProcess() and it'll initialise it with default values, and have a ObjectFactory::Singleton.Destroy(myObject) to destroy the object. Factories can keep count of the number of objects of different types that exist and return how much memory has been used by them. They're also useful if you want to allocate your object in a special location (e.g. from a predefined pool or a specific region of memory) other than where 'new' will allocate it.
Be creative, maybe you'll invent a new OO paradigm/abstraction and get to make a cool name for it

When you design your OS I recommend you find yourself a nice UML software package. UML is a modelling language (there is no code, the language IS the diagrams and stuff you draw) designed for OO software development. You can layout class diagrams so you can plan each class's members, and their relationships and stuff. You might find state diagrams useful as well.
You could inherit all of your managing classes (ProcessManager