<div>Hey,</div><div><br></div><div>So I was rereading my patch and remembered that I watch "graphics" for framebuffer devices, and then when udev finds them, I check to see if it's associated with a "drm" device:</div>
<div><br></div><div>+ /* We want to see if the framebuffer is associated with a DRM-capable</div><div>+ * graphics card */</div><div>+ udev = udev_device_get_udev (dev);</div><div>+ cards = udev_enumerate_new (udev);</div>
<div>+ udev_enumerate_add_match_parent (cards, udev_device_get_parent(dev));</div><div>+ udev_enumerate_add_match_subsystem (cards, "drm");</div><div>+</div><div>+ /* Udev assigns the "seat" tag to framebuffer and graphics card</div>
<div>+ * devices; without matching this tag, we also end up with the</div><div>+ * connected displays and the control interface to the card, which</div><div>+ * we don't care about at the moment. */</div>
<div>+ udev_enumerate_add_match_tag (cards, "seat");</div><div>+ udev_enumerate_scan_devices (cards);</div><div>+</div><div>+ card_entry = udev_enumerate_get_list_entry (cards);</div>
<div>+</div><div>+ if (card_entry != NULL)</div><div>+ {</div><div>+ struct udev_device *card;</div><div>+ const char *card_path;</div><div>+</div><div>+</div>
<div>+ card_path = udev_list_entry_get_name (card_entry);</div><div>+ card = udev_device_new_from_syspath (udev, card_path);</div><div>+ card_node = udev_device_get_devnode(card);</div>
<div>+</div><div>+ if (card_node != NULL)</div><div>+ {</div><div>+ ply_trace ("Found DRM card %s for framebuffer %s", card_node, fb_node);</div><div>+ card_node = strdup (card_node);</div>
<div>+ }</div><div>+ else</div><div>+ {</div><div>+ ply_trace ("This is strange, DRM card found with no device node...");</div><div>+ ply_trace ("Continuing as if no card found");</div>
<div>+ }</div><div>[snip]</div><div><div>+ ply_renderer_t *renderer = NULL;</div><div>+ </div><div>+ if (card_node != NULL)</div><div>+ {</div><div>
+ renderer = ply_renderer_new (PLYMOUTH_PLUGIN_PATH "renderers/drm.so",</div><div>+ card_node,</div><div>+ state->local_console_terminal);</div>
<div>+</div><div>+ if (renderer == NULL || !ply_renderer_open (renderer))</div><div>+ {</div><div>+ ply_trace ("Could not open renderer for %s, falling back to framebuffer", card_node);</div>
<div>+ ply_renderer_free (renderer);</div><div>+ renderer = NULL;</div><div>+ }</div><div>+ }</div><div>+</div><div>+ if (renderer == NULL)</div>
<div>+ {</div><div>+ renderer = ply_renderer_new (PLYMOUTH_PLUGIN_PATH "renderers/frame-buffer.so",</div><div>+ fb_node,</div><div>+ state->local_console_terminal);</div>
<div>+</div><div>+ if (renderer == NULL || !ply_renderer_open (renderer))</div><div>+ {</div><div>+ ply_trace ("Could not open renderer for %s, aborting", fb_node);</div>
<div>+ ply_renderer_free (renderer);</div><div>+</div><div>+ if (card_node != NULL)</div><div>+ {</div><div>+ free (card_node);</div>
<div>+ }</div><div>+</div><div>+ return;</div><div>+ }</div><div>+ }</div></div><div><br></div><div>This way we get automatic fallback to the framebuffer if the user's system doesn't have a drm-capable card. Do you think this is the right way to go about it?</div>
<div><br></div><div>Thanks,</div><div>Kevin</div><br><div class="gmail_quote">On Fri, Mar 1, 2013 at 11:16 AM, Ray Strode <span dir="ltr"><<a href="mailto:halfline@gmail.com" target="_blank">halfline@gmail.com</a>></span> wrote:<br>
<blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">Hi,<br>
<div class="im"><br>
> + udev_enum = udev_enumerate_new (state->udev);<br>
> + udev_enumerate_add_match_subsystem (udev_enum, "graphics");<br>
> + udev_enumerate_add_match_tag(udev_enum, "seat");<br>
> + udev_enumerate_add_match_is_initialized(udev_enum);<br>
> + udev_enumerate_scan_devices (udev_enum);<br>
> +<br>
> + entries = udev_enumerate_get_list_entry (udev_enum);<br>
><br>
> Unless I'm mistaken about how to drive the udev API, that should just simply<br>
> gather a list of all graphics devices with the "seat" tag that have been<br>
> fully initialized out of the list of all the devices that udev has detected<br>
> at that point; the ordering with framebuffer devices shouldn't come into<br>
> play.<br>
</div>I believe the "graphics" subsystem corresponds to /sys/class/graphics<br>
which is the legacy fb device subsystem.<br>
<div class="im"><br>
> As far as I remember, udev events don't fire until the device is fully initialized.<br>
<br>
</div>I remember looking into this a while back briefly. let me see if i<br>
can find the log... Ok it was actually from way back in october.<br>
Pasted below:<br>
<br>
<poettering> airlied: around?<br>
<poettering> ajax: or you?<br>
<poettering> got a question regarding fb and drm device nodes<br>
<poettering> do we get any guarantee about the order in which they<br>
show up in userspace?<br>
<poettering> the reason i am asking is this:<br>
<poettering> logind currently watches for fb device nodes to show up<br>
<poettering> it uses that as the main indicator for a seat being around<br>
<poettering> now, if the drm device node can show up after the fb node<br>
<poettering> then we'll inform gdm about the seat a bit too early<br>
<poettering> where it will start x at a time where drm is not yet available<br>
<poettering> now, what i am wondering about:<br>
<poettering> a) is there a guarantee that fb shows up after drm?<br>
<poettering> b) if not, can we get one?<br>
<poettering> c) if we can't get one, should i just forget about fb and<br>
always only watch drm?<br>
<poettering> d) and if c), are there any devices we stop supporting that way?<br>
<poettering> any other thoughts?<br>
<mjg59> poettering: Pretty sure that the drm device needs to exist for<br>
the KMS fb nodes to exist<br>
<mjg59> But that's not quite the same as the drm device *nodes* existing<br>
<mjg59> Let me look at that a bit more<br>
<poettering> to me only the order for the udev events matters really...<br>
<mjg59> poettering: This is the card node rather than any connectors, right?<br>
<mjg59> poettering: Looks like the card%d sysfs node will be created<br>
in drm_get_pci_dev()<br>
<mjg59> poettering: And I can't see any way the framebuffers could be<br>
created before that<br>
<halfline> well the real question is, at what point can we safely open<br>
/dev/dri/card0 and have it 1) exist and 2) not give any errors because<br>
we're trying to use it before things are fully initialized<br>
<ajax> mclasen: working on it, but afaik things should build in f18.<br>
<mclasen> ajax: ok, I'll try again<br>
<mclasen> ajax: also, did you see poetterings question ?<br>
<ajax> i did, i'm afraid i don't know that answer<br>
<ajax> my personal preference would be to not have fbdev emulation at<br>
all anymore really<br>
<ajax> but that probably doesn't help in the case of handing off the<br>
device from efifb to drm<br>
<mclasen> there seems to be a question about the readiness of<br>
/dev/dri/card0 too, though<br>
<ajax> that seems like a bad question though<br>
<mclasen> bad in the sense of ill-posed, or in the sense of 'you won't<br>
like the answer' ?<br>
<ajax> i guess i'm not sure why we'd get into a scenario where it<br>
could exist but be "not ready"<br>
<ajax> and if we could, why we wouldn't just consider that a drm bug.<br>
<mjg59> sysfs node is created before driver->load() is called<br>
<halfline> poettering,mjg59: looking at the code..drm_pci.c<br>
drm_get_minor(dev, &dev->primary, DRM_MINOR_LEGACY) creates the card0<br>
device and then dev->driver->load(dev, ent->driver_data) is called<br>
after taht<br>
<halfline> mjg59: jinx<br>
<ajax> uh, okay, let's just move A below B then.<br>
<ajax> nothing in ->load should be looking at the chardev info i don't think<br>
<mjg59> Yeah that does seem like a generally bad idea<br>
<mjg59> But does anything open card0 before there are any connectors?<br>
<halfline> well anyway ret = intel_fbdev_init(dev); is called after<br>
the whole lot<br>
<halfline> so what poettering is doing should be fine<br>
<halfline> assuming the other drivers are okay<br>
<ajax> mjg59: if we emit udev plug events when the connectors appear,<br>
then who cares?<br>
<ajax> just uevents i guess<br>
<halfline> well it's the difference between things occasionally<br>
failing and things not occasionally failing...<br>
<ajax> no, it's the difference between the opener being correct and<br>
not being correct.<br>
<halfline> heh<br>
<ajax> connectors can already come and go<br>
<ajax> so you ought already to be looking for that<br>
<halfline> ought but not<br>
<halfline> (for plymouth anyway)<br>
<ajax> yeah, i didn't say we _are_ correct, just that the way forward<br>
is, er, straightforward<br>
<ajax> but in the static connector list case i think those all exist<br>
by the time ->load is done<br>
<mclasen> makes sense; if there are events for connectors appearing,<br>
we should deal with them<br>
<ajax> so simply not exposing the chardev before initialization seems<br>
like it'll paper over that ordering issue<br>
<mjg59> ajax: As long as ->load() doesn't create any connector objects<br>
<mjg59> Because that depends on the sysfs parent already existing<br>
<ajax> clearly i'm missing something here<br>
<halfline> well poettering is just waiting for the /dev/fb device to<br>
show up which should be fine paper<br>
<mjg59> ajax: drm_get_minor() is currently called before the driver<br>
->load() method. This guarantees that there's a sysfs object for the<br>
card, and any connectors will use that as their parent. You suggested<br>
moving it to after ->load(). That's fine as long as nothing tries to<br>
create connectors in the load() method.<br>
<ajax> remind me which cog in this hell machine is responsible for<br>
making the /dev node exist<br>
<mjg59> ajax: udev will create the device node some amount of time<br>
after the sysfs object is created<br>
<mjg59> ajax: But depending on it to not do that until after load() is racy<br>
<mjg59> It could happen at any time after the card sysfs object is created<br>
<ajax> this seems like a class of races that others have to have faced<br>
<mjg59> I blame drm's awful abstractions<br>
<mjg59> Anyway, the device will normally be created during boot, so<br>
the only thing that could really be racing with it is plymouth<br>
<mjg59> Lennart's stuff is waiting for framebuffer creation, which<br>
works as a useful extra sequencing point<br>
<halfline> cosimoc's machine boots so fast that plymouth isn't in the<br>
picture and he gets software rendering for gnome-shell<br>
<halfline> well once in a while<br>
<halfline> because pam_systemd runs before the /dev/dri/card0 exists<br>
<halfline> and so it doesn't given gdm access rights to it<br>
<halfline> so we're going to make GDM not start an X server until<br>
systemd gives the go ahead<br>
<halfline> and systemd gives the go ahead when /dev/fb0 shows up<br>
<halfline> so we should be good<br>
<poettering> mjg59: so you say fb is a good sync point?<br>
<poettering> if that's the case then i am happy<br>
<poettering> and if fb is dropped one day we can reinvestigate the whole issue<br>
<poettering> all i am aksing for basically is that by the time fb was<br>
handled by userspace all initialization that X might need of the<br>
drivers is complete<br>
<halfline> poettering: so i checked intel, radeon, and nouveau<br>
earlier, and they all did fbdev initialization at the end<br>
<halfline> so i think you're safe<br>
<mjg59> poettering: "Good" may be overstating<br>
<mjg59> But I suspect it'd work<br>
<notting> does gnome-tweak-tool tweak non-dconf settings?<br>
<kay> mjg59: udev does not create any device node since quite some<br>
time, it's all in the kernel, /dev is at the same time as /sys<br>
<mjg59> kay: Yeah ok I need to pay more attention<br>
<kay> mjg59: udev today does only symlinks and permissions at event<br>
time. devtmpfs does all the nodes stuff. udev does not even have the<br>
code anymore to call mknod() in the hotplug handler ...<br>
<airlied> please don'<br>
<airlied> please don'y rely on /dev/fb nodes<br>
<airlied> its not good<br>
<airlied> all the other stuff is protected by drm_global_muetx<br>
<airlied> so if you open before we area ready it'll block<br>
<airlied> since drm_stub_open also takes the drm_global_mutex<br>
<airlied> we don't want to require drm to provide fb emulation for ever<br>
<airlied> poettering, halfline, ajax, mjg59 ^<br>
<halfline> airlied: hmm, pretty sure we've had bugs plymouth bugs in<br>
the past where if we open /dev/dri/card0 too early it's there but<br>
fails<br>
<halfline> anyway the problem this time around was it not being there at all<br>
<halfline> so pam_systemd couldn't trigger ACL changes to it<br>
<airlied> halfline: yeah there was problems years ago<br>
<airlied> its almsot like we fixed them :-)<br>
<airlied> though I'm trying to remember when we did<br>
<airlied> b64c115eb22516ecd187c74ad6de3f1693f1dc7b 2 years ago<br>
<airlied> I've had alternate patches to move stuff around but I never<br>
trusted them enough<br>
<airlied> and I didn't think open was too contended<br>
</blockquote></div><br>