Page 1 of 1

Design of data-descriptor for code in ring 1 or 2

Posted: Fri May 16, 2014 11:47 am
by edonner
Hi folks,

I wonder about how to design the data-descriptor for a “module” (driver, executable) in rings 1 and 2 and like to know your experiences or hints because maybe I’m totally wrong and lost.

The virtual memory of my OS defines the first virtual GB for use in ring 0-2, the virtual memory above is for ring 3. A task-change shall change the memory-mapping just for ring 3 (above 1GB), the first virtual GB remains the same in all tasks.
Now I do a call from ring 3 to ring 2. The SS changes regarding to my TSS, the DS doesn’t change. Now I’m in ring 2 with the DS of ring 3.

The DS of ring 3 points to a “user-mode-descriptor” using virtual memory above 1GB. I’m pretty sure that my ring-2-code might need to access data from this user-space for whatever reason. But I’m pretty sure too that my ring-2-code also need access to “local data” not in user-space but in the first virtual GB. Unfortunately, with the actual DS from user-space, the ring-2-code cannot access virtual memory in the first GB. Right?

So I think about changing the DS in my ring-2-code to use another descriptor. And now I don’t know how this descriptor should look like. I have two ideas, both I don’t like:

(1) Every “module” gets its own data-descriptor and describes exactly the data-space the module needs. I think the positive thing is that this is a safe memory-model. No driver/module could write into memory of another module. The disadvantages are a lot of work for the virtual manager if the module calls “malloc()”: I might need to reserve new virtual memory, copy the old mem into the new mem and adjust the DS-descriptor. Even its hard to access user-space: I would need to use a 2nd descriptor (ES) to access the user-memory (I code in C++, so using just one “DS” would be comfortable)

(2) Every “module” has access to virtual memory from 0 to FFFFFFFF. When a module is called, I don’t need to change DS. When I access mem in the 1st GB, I access “kernel data” independent on the current task and I can access user-data w/o any problems. But the big disadvantage: When my module is buggy and writes to whatever memory in the 1st GB, it is able to destroy all kernel/OS data because it is able to write into the virtual mem of all other modules.
How do you handle this? Are there other options? Am I basically wrong?

Thanks a lot,
Enrico.

Re: Design of data-descriptor for code in ring 1 or 2

Posted: Fri May 16, 2014 12:19 pm
by Combuster
Any particular reason why you're not using paging to solve the problems you have?

Re: Design of data-descriptor for code in ring 1 or 2

Posted: Fri May 16, 2014 1:23 pm
by edonner
Hi Combuster,

no, probably the simple reason is that I need to know more and I'm not experienced. I'm just a SAP-Programmer, trying to do some OS-development over the years to keep my brain busy ...

I just wanted to change any paging-data during a task-change. I dont want to change paging-data when changing protection-levels. Why? Because I'm afraid it's too expensive. I could imagine I get a very safe code when I run each module it it's own task. Instead of a call-gate, I use a task-gate and change all page-mapping so that my "module" just see's what it shall see. But ... too expensive and strange, isn't it?

I guess you give me a good hint with your question but to be true, I dont see the story behind yet. Need to think about it.

Anyway, thank you and I hope you guys can help a simple ABAP-programmer to continue on his OS :-)
Enrico.

Re: Design of data-descriptor for code in ring 1 or 2

Posted: Fri May 16, 2014 3:18 pm
by Combuster
Mixing protection mechanisms is an art on it's own right. The reason most people use only paging is because the basic implementations are fairly simple. At the very least, using task gates is a big performance hit, doesn't allow you to do administration on task switches, and causes all sorts of technical implementation issues.

The typical implementation of mixing segmentation with paging is to give applications only a limited subset of the address space to use, giving you a number of applications to switch between without having to switch page tables. It also allows you to do allocations without needing to copy chunks of data.

Re: Design of data-descriptor for code in ring 1 or 2

Posted: Sat May 17, 2014 1:07 am
by edonner
Does it mean you suggest loading a new CR3 when calling to a different module?

Doing so I could give an own adress-space to every module, even from 0 to FFFFFFFF.
"malloc()" could be done quite easy by assign new pages into the virtual Memory space.

Hm yes, I was thinking of this before but for me this sounds very similar to a task-change in terms of costs.
I wanted to avoid changing the paging data because of the time (I suppose) it consumes ...