Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

x86 PICs as devices #1457

Open
wants to merge 72 commits into
base: main
Choose a base branch
from
Open

x86 PICs as devices #1457

wants to merge 72 commits into from

Conversation

ehem
Copy link
Contributor

@ehem ehem commented Oct 9, 2024

A major reason behind non-x86 PICs being devices is to utilize the module and newbus infrastructure. This does make x86 the odd one out, so turning the x86 PICs into full devices will likely aid converging the interrupt systems.

Having PICs for all architectures be proper devices allows the interrupt event core use device methods for interrupt callbacks. This shrinks the interrupt event data structures as everything can be done via a single pointer, rather than a pointer for each callback.

If interrupt events could be embedded inside the architecture interrupt structures, struct intr_event could be further shrunk by removing another pointer. The result would make the architecture core <=> event core relationship similar to the PIC <=> architecture core relationship.

@ehem ehem force-pushed the x86picdev branch 6 times, most recently from de5428d to 7ae23df Compare October 14, 2024 20:19
@ehem ehem closed this Oct 14, 2024
@ehem ehem deleted the x86picdev branch October 14, 2024 20:19
@ehem ehem restored the x86picdev branch October 14, 2024 20:19
@ehem ehem reopened this Oct 14, 2024
@ehem ehem marked this pull request as ready for review October 14, 2024 20:31
@ehem ehem requested review from bsdimp and bsdjhb as code owners October 14, 2024 20:31
@ehem
Copy link
Contributor Author

ehem commented Oct 14, 2024

This has ended up going rather further than I had expected before terminating.

The old Marvell GPIO driver has needed a severe update for some time. It has become so rotten I can no longer work around the problems.

The x86 local-APIC and IO-APIC drivers have been tested. The AT PIC driver may work, but I haven't yet gotten a good recipe for qemu. The impact on x86/Xen and x86/MSI drivers is small enough for them to likely still be working, but they're not yet tested.

@ehem
Copy link
Contributor Author

ehem commented Oct 16, 2024

This is certainly going to need some adjustment, but I need advice as to which direction to go with this. There are 3 places I see where this series could be reasonably terminated.

First, it could be terminated after converting the x86 PICs to devices. This was the main goal. This fails to pick up most of the benefits, but does make the interrupt infrastructures consistent in PICs being devices.

Second, it could terminate after the conversion to int intr_event_create_device(struct intr_event **event, device_t pic, interrupt_t *source, u_int irq, int flags, const char *fmt, ...). This shrinks the structures for hardware interrupts by 24 bytes (3 fewer pointers) and enlarges software interrupts by 16 bytes. Many systems will have >100 hardware interrupts and with 3-7 software interrupts being the maximum, this is a notable benefit. This requires conversion to #include <machine/interrupt.h> in many places.

Third, it could terminate after converting to int intr_event_init(struct intr_event *ie, device_t pic, u_int irq, int flags, const char *fmt, ...). I feel having the PIC <=> arch core interface being similar to the arch core <=> shared interface is a good thing. I've found it rather bizarre how the PIC structure wraps the architecture core, but the architecture core then has a pointer to the shared core. Not having memory allocation in intr_event_init() allows the architectures to initialize their intr_event structures sooner as they can be setup while the appropriate mutex is held. This also reduces memory usage for hardware interrupts by 48 bytes and enlarges software interrupts by 8 bytes.

Overall this seems a rather good idea. Yet now I'm waiting on reviewers...

@bsdimp
Copy link
Member

bsdimp commented Oct 22, 2024

This change is too large. It has a lot of gratuitous changes that aren't strictly necessary. 49 changes and 244 files is simply too much to review. A quick survey of the changes shows lots of header file renames (which aren't needed) and kobj function name renaming (also need needed). In it's current form, it's not possible to review this.

@ehem
Copy link
Contributor Author

ehem commented Oct 24, 2024

That is what happens if you look at a squash of 49 commits. Take almost any sequence of 49 commits in FreeBSD's repository, look at the delta and it will be confusing. There is also some infrastructure which has been waiting 2.5 years and had that been brought in in a timely fashion it wouldn't need to be included here.

For some overview of the earlier ones, 5a4854b ended up getting in via other means. It is here as I was expecting it to be brought in with this, but then unexpectedly it came in via other means.

The next 3 are 2 bits of cleanup and one thing I've got an unpleasant suspicion will explode in the not too distant future.

Commit #5 is a key commit. Without that the approach is impossible. Notably x86 presently initializes PICs at SI_SUB_INTR while newbus was doing all initialization at SI_SUB_DRIVERS. Initializing the root bus earlier works fine, but I'm unsure whether that will be accepted.

Advice on the question asked immediately above might allow abandoning some pieces. Might be clearer to point to the primary commit for intr_event_create_device(). The point is the arguments to intr_event_create() are rarely independent. For any given PIC, the functions will always be the same. As such if the interface moves to intr_event there is substantial savings on hardware interrupts since only a single pointer is used.

There isn't a good commit to point to for intr_event_init(), though perhaps f2abf73 can give some idea. I've long been bothered by how INTRNG and x86 PICs wrap their data around the architecture interrupt type, then cast back and forth. Whereas both INTRNG and x86 include a struct intr_event *<prefix>_event; inside their respective structures.

Why is one boundary embedding the structure inside the other, while the other is handled via pointer? These two relationships are similar, so shouldn't their interfaces be similar? Embedding the structure results in 16 bytes of savings since two pointers are removed (pointer to intr_event and there is no longer any need for ->ie_source).

@ehem ehem force-pushed the x86picdev branch 3 times, most recently from c233398 to b7da32e Compare October 31, 2024 05:13
sys/kern/subr_bus.c Outdated Show resolved Hide resolved
sys/kern/subr_bus.c Outdated Show resolved Hide resolved
sys/kern/subr_bus.c Outdated Show resolved Hide resolved
sys/sys/bus.h Outdated Show resolved Hide resolved
sys/kern/subr_bus.c Outdated Show resolved Hide resolved
@ehem ehem force-pushed the x86picdev branch 2 times, most recently from ac8444c to 8398309 Compare November 1, 2024 17:48
@@ -5108,7 +5109,6 @@ root_bus_module_handler(module_t mod, int what, void* arg)
{
switch (what) {
case MOD_LOAD:
TAILQ_INIT(&bus_data_devices);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This change doesn't move the needle enough to take at this time. bus_data_devices is the first thing that does this, so that makes this a little inconsistent. It's fine how it is, and there's not a compelling reason to do this. Please drop this change from this commit series.

Copy link
Contributor Author

@ehem ehem Nov 4, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I had been considering squashing it into the move to => SI_SUB_INTR. Mainly I had looking at what could be pulled out of root_bus_module_handler().

sys/kern/subr_bus.c Outdated Show resolved Hide resolved
ehem added 29 commits January 15, 2025 12:00
As all x86 PICs must set this value, it seems more consistent to do so
inside intr_register_source().  Hopefully a trivial reduction in work.
Now that the value is on the event, there is no need for x86 to
duplicate the value on the intsrc structure.
These were needed for the struct->device transition.  Now purge these
as they are no longer needed.
The two remaining uses of `atpic_vector()` were both in
`atpic_config_intr()`.  While not a super-expensive function, reusing
the cached value is faster.  More notably this is a mild simplication of
the file.

Inline the IRQ() macro as that now only has a single use.

Note that the vector variable is never changed.
Using mtx_owned() should result in slight stack shrinkage.  The use of
the locked state is a less commonly encountered state anyway.
As empty pic_suspend hooks are accepted as NOP, simply remove this
unnecessary function/pointer.
Convert large numbers of instances of overt driver_t structures to using
the DEFINE_CLASSN() macros.  Greater consistency should reduce future
errors.  Convert several uses of '{ 0, 0 }' driver method list ends to
DEVMETHOD_END macro.

Replace uses of `static` before `DEFINE_CLASS_0` with
`PRIVATE_DEFINE_CLASSN()`.

Remove instances of commas after "END" macros.  Ensure empty lines
before DEVMETHOD_END macros.
This allows the interrupt framework to introduce hooks into device
function tables.  Some functionality may be common and the framework
may provide standard functionality.  Certain PIC tasks may also go
through the function tables.
This reduces the memory consumed by the intr_event structures.  The
hooks associated with any given PIC don't vary.
…ndling

As the interrupt event interface is friendly to all INTRNG drivers,
switch to that.  Remove the INTRNG version of the interface as it is now
redundant.  Yet to be introduced drivers are likely to need updates.

Leave base class implementation behind for debugging.
These were global, but now appear unused outside the file.  As such now
mark them static, mostly to document their current status.
These aren't flaged as unused due to `mv_gpio_setup_intrhandler()` being
global.  They don't appear to be used anywhere though, so completely
disable them.

I can only confirm `->gpio_events` being set by the above.  As such the
references to the variable would effectively be NOPs.

I suspect this could be used by something outside of the FreeBSD
tree, but with no references I'm unable to confirm this.
This driver was never fully adapted to work with INTRNG (current ARM
interrupt core).  Since the driver needs a full rework at this point,
disconnect the driver from the FreeBSD kernel build.  Hopefully this
might be updated by someone, but it has become too problematic.
The intr_event structure is a baseline for what all architectures
support.  This isn't truly separate from the infrastructures and is
always allocated.  As such merging seems best.
Now that the value is on the event, there is no need for INTRNG to
to duplicate the value on the intr_irqsrc structure.
Always allocating `intr_event` means we have a better place to store
the interrupt number.  Implement using this in all cases.
FreeBSD/ARM is now always INTRNG, therefore these are no longer needed.
Simple cleanup which ideally would have been done some time ago.
Convert large numbers of instances of overt driver_t structures to using
the DEFINE_CLASSN() macro.  Greater consistency should reduce future
errors.  Convert several uses of '{ 0, 0 }' driver method list ends to
DEVMETHOD_END macro.

Small amounts of similar cleanup to similar macros.  The opal_sensor
module had omitted the sentinal from the table.  That had likely been
functioning due to dumb luck for some time.

Differential Revision: https://reviews.freebsd.org/D47546
Spotted while examing this driver.  Remove some end of line spaces/tabs.
Using kobj allows implementing most PIC functions merely by inheriting
from the parent class.  As there are multiple OpenPIC implementations,
this ensures all common hooks go through by default.
This finishes removing OpenPIC functions from the global namespace.
Instead PICs based on OpenPIC should use `openpic_class` for explicitly
calling OpenPIC functions.

Filter out a few end of line spaces while at it.
The correct behavior for cascading PICs is to pass the parent trapframe.
This allows the interrupt framework to introduce hooks into device
function tables.  Some functionality may be common and the framework
may provide standard functionality.  Certain PIC tasks may also go
through the function tables.

In the case of PowerPC there need to be two classes.  The core includes
extra functionality not present on other platforms.  As such
"pic_hw_class" matches the existing use which assumes hardware PICs.  In
particular the core handles level/edge, low/high and passes those to the
PIC.  The PIC then has a private area pointer.  Whereas other
architectures the PIC handles the interrupt parameters for itself.

Also include a "pic_base_class" for a future where this was split off.
Create an IPI device for handling IPI interrupts.  While modifying the
internals of an interrupt event is concerning, this does avoid the
present troubles from 0c50edf.  ->ie_pic may well replace the
architecture counterparts on machine interrupt source structures, at
which point the value would be non-removable anyway.
This reduces the memory consumed by the intr_event structures.  The
hooks associated with any given PIC don't vary.
…tructures

The intr_event structure is a baseline for what all architectures
support.  This isn't truly separate from the infrastructures and is
always allocated.  As such merging seems best.

As `intr_event_init()` does not sleep, it can be called while locks are
held.  This allows cleanup of `intr_lookup()` as the event can
be initialized while the lock is held.  The removes the lazy event
allocation, pushing architectures closer together.
Now that the value is on the event, there is no need for PowerPC to
duplicate the value on the powerpc_intr structure.
Each architecture can convert to kobj interrupt interface independently.
This is the final result of all architectures fully implementing use of
the kobj interrupt interface.
As all FreeBSD architectures provide a PIC base class, move the
declaration to the common header.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants