Stream: brlcad

Topic: globals in libdm


view this post on Zulip starseeker (Apr 09 2020 at 20:10):

@Sean Question - are the globals in the display manager backends (like, for example, ambientColor in dm-ogl.c) intended to be global across all display managers? I'm trying to figure out why they would have been included in the local variables associated with a given display manager...

view this post on Zulip Sean (Apr 09 2020 at 20:11):

looking

view this post on Zulip starseeker (Apr 09 2020 at 20:11):

er - would NOT have been included in the local dm variables, sorry

view this post on Zulip Sean (Apr 09 2020 at 20:13):

No, they're not intended to be global across all.

view this post on Zulip starseeker (Apr 09 2020 at 20:14):

OK, thanks - that establishes some degrees of freedom

view this post on Zulip Sean (Apr 09 2020 at 20:14):

they're a rather specific implementation detail of opengl 2

view this post on Zulip Sean (Apr 09 2020 at 20:14):

so you'll probably see them replicated in all the DMs using opengl, and not elsewhere

view this post on Zulip starseeker (Apr 09 2020 at 20:14):

Ah, I wasn't clear - I meant whether they're intended to be global across all instances of display managers of a specific type

view this post on Zulip Sean (Apr 09 2020 at 20:15):

if we made a vulkan or ogl4 interface, some of them would likely disappear or change

view this post on Zulip starseeker (Apr 09 2020 at 20:15):

so if you set ambientColor on one ogl dm, is it expected that that setting be reflected across all ogl instances?

view this post on Zulip Sean (Apr 09 2020 at 20:15):

ah, hm

view this post on Zulip starseeker (Apr 09 2020 at 20:15):

icky if it does...

view this post on Zulip Sean (Apr 09 2020 at 20:15):

probably yes

view this post on Zulip Sean (Apr 09 2020 at 20:16):

that's the only reason to make them static

view this post on Zulip starseeker (Apr 09 2020 at 20:16):

<wince>

view this post on Zulip Sean (Apr 09 2020 at 20:16):

C version of class data

view this post on Zulip starseeker (Apr 09 2020 at 20:17):

That's kinda not cool - what if the user application doesn't want the same settings in all windows??

view this post on Zulip Sean (Apr 09 2020 at 20:20):

they don't appear to be settings in the same sense of DM settings, but I don't feel strongly about it

view this post on Zulip starseeker (Apr 09 2020 at 20:20):

Oh, well - I guess compared to the rest of this that's a minor pain point...

view this post on Zulip Sean (Apr 09 2020 at 20:20):

they're more "these are values we have to tell opengl for things to display" that could just as well be static inside of functions except that several functions need that value

view this post on Zulip Sean (Apr 09 2020 at 20:21):

like position of the light source -- that's completely arbitrary. just needs to be defined.

view this post on Zulip starseeker (Apr 09 2020 at 20:21):

So rather than pull the values from the private vars struct, they're just stashed in statics?

view this post on Zulip starseeker (Apr 09 2020 at 20:22):

I suppose it doesn't really matter, just a bit odd when reading a function and a variable swoops in from nowhere...

view this post on Zulip Sean (Apr 09 2020 at 20:23):

Just looking through the code, I would say the main distinction looks to be more that the statics are reflective values that just get set and/or updated as needed, whereas the private vars are declarative values that control and modify

view this post on Zulip Sean (Apr 09 2020 at 20:25):

If this were a C++ class, it's the difference between class data and object data

view this post on Zulip starseeker (Apr 09 2020 at 20:25):

How they intruded on my radar screen - I was experimenting with moving the *_open calls (which are relatively heavy Tcl/Tk users) up into libtclcad and tripped on the statics.

view this post on Zulip Sean (Apr 09 2020 at 20:25):

class { static int var = val; } vs class { int var = val; }

view this post on Zulip starseeker (Apr 09 2020 at 20:25):

I suppose that's a fairly hare-brained idea to begin with

view this post on Zulip Sean (Apr 09 2020 at 20:28):

how so? looks like they only set dm_interp and don't actually do anything with it

view this post on Zulip starseeker (Apr 09 2020 at 20:29):

dm_plot at least set the statics in its open call. I went ahead and stuffed those into the plot_vars, but I haven't audited the more complex globals to see if the _open functions monkey with them or not.

view this post on Zulip Sean (Apr 09 2020 at 20:30):

but yeah, conceptually, that makes no sense... a DM API without a way to open a DM would be broken design-wise.. wouldn't do it just for the sake of moving a type.

view this post on Zulip starseeker (Apr 09 2020 at 20:31):

I'm trying to get Tcl/Tk out of libdm and libfb - I'm pretty sure I've gotten libged itself clear now, so the dm and fb layers are the last holdouts.

view this post on Zulip Sean (Apr 09 2020 at 20:31):

you know you're in super-likely-to-change-behavior territory... ? :)

view this post on Zulip Sean (Apr 09 2020 at 20:31):

and by 'change', I mean introduce a bug ;)

view this post on Zulip starseeker (Apr 09 2020 at 20:31):

Yeah, that's why I had to revert my first attempt - worked on Linux, busted on Windows

view this post on Zulip Sean (Apr 09 2020 at 20:31):

it's all good, just beware :)

view this post on Zulip starseeker (Apr 09 2020 at 20:32):

There's a lot of really weird black magic going on with X, ogl, wgl, and Tk in this - it's incredibly fragile

view this post on Zulip starseeker (Apr 09 2020 at 20:33):

I'll scrub down the verbose "cast everything at every usage even though the cast is 20 chars long" style to try and make it more readable (THAT at least should be safe enough), but this is one intimidating maze...

view this post on Zulip Sean (Apr 09 2020 at 20:34):

such as? they've always looked pretty CG101 to me. it's not black magic, it's just what one has to do

view this post on Zulip Sean (Apr 09 2020 at 20:34):

like the ogl DM is pretty text book

view this post on Zulip Sean (Apr 09 2020 at 20:35):

most of the ugly is ... the X11 api and that's just how it is

view this post on Zulip starseeker (Apr 09 2020 at 20:35):

The platform specific voodoo for embedding the Tk window, trying to get it to repaint at the right times, the platform specific differences beyond glx vs wgl despite Tk nominally being cross platform...

view this post on Zulip starseeker (Apr 09 2020 at 20:36):

I know a lot of this is Tk's fault for not providing a proper OpenGL widget, but that doesn't make it any less annoying to deal with.

view this post on Zulip Sean (Apr 09 2020 at 20:37):

yeah, that's just an artifact at this level of API. DM creates the lowest level contexts and needs a handle on them, as does Tk, so the two have to cooperate somehow

view this post on Zulip starseeker (Apr 09 2020 at 20:38):

From a design standpoint that's what's binding libged et. al. to Tk right now though - that low level connection. Just conceptually, what should a libged+dm+fb functionality set look like without Tk in the mix?

view this post on Zulip Sean (Apr 09 2020 at 20:38):

as it is, Tk is being rather polite in that we can do the init and hand it our context. lots of APIs don't make that work, especially higher level apis like game engines.

view this post on Zulip starseeker (Apr 09 2020 at 20:39):

<snort> they'd better be polite about it, if they're not going to do the right thing and give us a properly integrated widget to begin with

view this post on Zulip Sean (Apr 09 2020 at 20:40):

right, the alternative is usually fully managed where they don't let you get at the context

view this post on Zulip Sean (Apr 09 2020 at 20:41):

from libged's perspective, it needs the original dm/fb callback API design -- you just open a dm and draw into it.

view this post on Zulip Sean (Apr 09 2020 at 20:42):

if everything is properly defined through the callbacks, ged should be just fine without having any awareness of Tk

view this post on Zulip starseeker (Apr 09 2020 at 20:42):

Hmm... I think the explicit linkage right now is with fb, actually...

view this post on Zulip starseeker (Apr 09 2020 at 20:44):

So would we define a lightweight/high-level dm/fb API that is what libged uses, and then it's the responsibility of libdm/lifb/whatever else to implement that minimal API?

view this post on Zulip Sean (Apr 09 2020 at 20:45):

I was going to say, maybe I've already updated and you've cleared it out, but libdm doesn't really seem to do much at all with Tk or Tcl

view this post on Zulip starseeker (Apr 09 2020 at 20:45):

Um. The details of the dm-ogl and dm-X backends do...

view this post on Zulip Sean (Apr 09 2020 at 20:45):

the biggest issue with fb last time I looked is reconciling what open_existing is doing

view this post on Zulip Sean (Apr 09 2020 at 20:46):

starseeker said:

So would we define a lightweight/high-level dm/fb API that is what libged uses, and then it's the responsibility of libdm/lifb/whatever else to implement that minimal API?

Yes, that was the original design of them both too.

view this post on Zulip starseeker (Apr 09 2020 at 20:47):

I had to revert my attempt to push Tk out of the backends when it broke on Windows (frustrating, I was actually fairly close, but debugging that sort of problem on Windows could be a long haul). From what you were saying that probably wasn't the right way to go at it anyway.

view this post on Zulip Sean (Apr 09 2020 at 20:48):

hence all the #define DM_* ... defines that should all generically hook into the dmp callback table for a given dm instance

view this post on Zulip Sean (Apr 09 2020 at 20:49):

starseeker said:

Um. The details of the dm-ogl and dm-X backends do...

example? line?

view this post on Zulip starseeker (Apr 09 2020 at 20:49):

dm_ogl.c:569

view this post on Zulip starseeker (Apr 09 2020 at 20:50):

dm_ogl.c:632 - basically just search for Tk_ in the file

view this post on Zulip Sean (Apr 09 2020 at 20:53):

ah, well yeah but that's the actual window. it needs a window to draw into and one doesn't exist so it creates it, initializes .. but it doesn't really "do" anything with it that results in a back and forth with the Tcl interpreter

view this post on Zulip starseeker (Apr 09 2020 at 20:54):

Sure. But the necessity of calling all those Tk_ functions binds us to the Tcl/Tk libraries below the libged level.

view this post on Zulip Sean (Apr 09 2020 at 20:54):

probably being imprecise with my words, sorry -- dm presently does have the responsibility for create a window

view this post on Zulip Sean (Apr 09 2020 at 20:54):

to date, that has been (and needed to be) a Tk window

view this post on Zulip starseeker (Apr 09 2020 at 20:55):

OK. Then as long as that's true and we need Tk windows, we're stuck unless we switch to a gcv style "load the backend as a plugin" setup and have libtclcad build the backends...

view this post on Zulip starseeker (Apr 09 2020 at 20:56):

In some ways it doesn't look to far away from that, actually...

view this post on Zulip Sean (Apr 09 2020 at 20:56):

right, you're not going to change that binding moving it without either changing dm responsibility or changing (or perhaps violating) architecture in some manner.

view this post on Zulip starseeker (Apr 09 2020 at 20:58):

OK. So, path forward?

view this post on Zulip Sean (Apr 09 2020 at 20:58):

I was going to say, a workable solution to avoid symbol linkage would be to either do dynamic loading or change the responsibility

view this post on Zulip Sean (Apr 09 2020 at 20:58):

what exactly is the goal? to simply remove Tk as a linkage yet still keep creating Tk windows?

view this post on Zulip Sean (Apr 09 2020 at 20:58):

that wouldn't seem incredibly useful to me

view this post on Zulip Sean (Apr 09 2020 at 20:59):

to set the stage for another window mechanism?

view this post on Zulip starseeker (Apr 09 2020 at 20:59):

Well, my goal is libged capabilities without the necessity of Tcl/Tk - for example, if I were to define a GLFW based display for gsh I would want to be able do at least do old-style console display managers and frame buffers.

view this post on Zulip starseeker (Apr 09 2020 at 21:01):

If I can do that, then the stage should be properly set for a Qt interface

view this post on Zulip Sean (Apr 09 2020 at 21:02):

so devil is definitely in the details to what you just said

view this post on Zulip Sean (Apr 09 2020 at 21:02):

to do what you describe and what I think you're getting at, I think we'll need to separate the concept of window and context creation in the DM api

view this post on Zulip Sean (Apr 09 2020 at 21:02):

right now, they are one in the same throughout dm

view this post on Zulip Sean (Apr 09 2020 at 21:04):

the API isn't far off, but it needs to be one layer lower on the open/close calls and a layer higher on everything else

view this post on Zulip Sean (Apr 09 2020 at 21:06):

for example, currently, dm_open does both -- it creates a window and sets up a context. dm_open_existing avoids both I believe, which is why there is a platform version of each.

view this post on Zulip Sean (Apr 09 2020 at 21:07):

to be able to utilize a window and/or context that someone else creates, things WILL get messy and likely platform-specific in application code but it can work

view this post on Zulip starseeker (Apr 09 2020 at 21:08):

Hmm. So we'll have to basically sketch out a reworked dm/fb API, then identify an implementation/migration path from where we are to it?

view this post on Zulip Sean (Apr 09 2020 at 21:09):

not the only option, but that's a possibility sure

view this post on Zulip starseeker (Apr 09 2020 at 21:09):

What if we make the libdm backends into plugins, keeping the window creation where it is, but allowing a program to provide its own custom backend?

view this post on Zulip Sean (Apr 09 2020 at 21:12):

Decorator Pattern: https://en.wikipedia.org/wiki/Decorator_pattern

view this post on Zulip starseeker (Apr 09 2020 at 21:13):

Am I correct that's more or less what we envision for libgcv?

view this post on Zulip Sean (Apr 09 2020 at 21:13):

don't need to be plugins per se, they could be, but the important aspect is that you have basically two apis -- a front end and a backend

view this post on Zulip starseeker (Apr 09 2020 at 21:14):

I guess I would need to whiteboard it - my brain jumps to the dmp->dm_* calls as being the front end and the X/ogl/wgl specific implementations being the back end, but I know I'm missing the real point...

view this post on Zulip Sean (Apr 09 2020 at 21:15):

dm is presently concerned with front+back simultaneously, thus dm-ogl.c is really X11 user frontend and OpenGL drawing backend; dm-X.c is X11 user frontend and X11 drawing backend; dm-wgl is WGL frontend and OpenGL drawing backend, etc

view this post on Zulip starseeker (Apr 09 2020 at 21:15):

Oh, by frontend you're referring to the Windowing API?

view this post on Zulip Sean (Apr 09 2020 at 21:15):

it's like a driver interface, rather it needs to be that

view this post on Zulip Sean (Apr 09 2020 at 21:16):

think of it this way, look at dm-ogl.c and think what it'd take to extricate OpenGL

view this post on Zulip Sean (Apr 09 2020 at 21:18):

similar for the window creation calls (e.g., i.e. the Tk+X11 calls)

view this post on Zulip starseeker (Apr 09 2020 at 21:18):

Hmm. So, create then Window and then use that to inform the creation of the context?

view this post on Zulip Sean (Apr 09 2020 at 21:19):

no, the become separate concepts

view this post on Zulip Sean (Apr 09 2020 at 21:20):

this direction would actually greatly benefit from C++... sigh

view this post on Zulip Sean (Apr 09 2020 at 21:21):

it'd be something like dm_create_window(), dm_create_context(), dm_register_context()

view this post on Zulip starseeker (Apr 09 2020 at 21:22):

Ah, OK - I hadn't succeeded in parsing the code well enough yet to have a sense that you could create the context without the window.

view this post on Zulip Sean (Apr 09 2020 at 21:26):

that all said, I would suggest focusing on just the minimal necessary bits that simply encapsulate all the Tcl/Tk calls and whatever else is necessary so interp is not part of the public API

view this post on Zulip Sean (Apr 09 2020 at 21:28):

that's a GREATLY reduced subset of work needed to make that happen. you can't/shouldn't pull Tk out of DM without surgery, but GED doesn't need to know anything whatsoever about the Tcl/Tk in DM

view this post on Zulip Sean (Apr 09 2020 at 21:28):

that'll be good forward progress even if we later burn it all to the ground

view this post on Zulip Sean (Apr 09 2020 at 21:29):

I'd probably start with interp

view this post on Zulip starseeker (Apr 09 2020 at 21:29):

I think the only instances of Tcl/Tk in the public API (at least, per the headers) are the interp and he dm/dm_xvars.h Tk_Windows.

view this post on Zulip Sean (Apr 09 2020 at 21:29):

yep

view this post on Zulip starseeker (Apr 09 2020 at 21:30):

IIRC the only overt use I saw of interp was in wgl - something about deiconifying the window.

view this post on Zulip Sean (Apr 09 2020 at 21:30):

figure out how to hide interp, and you'll be half done

view this post on Zulip Sean (Apr 09 2020 at 21:30):

interp is used to create tkwin via Tk_MainWindow(interp)

view this post on Zulip starseeker (Apr 09 2020 at 21:30):

Oh, the other one was the _init_dm call.

view this post on Zulip Sean (Apr 09 2020 at 21:31):

interp kicks of all the tk calls

view this post on Zulip starseeker (Apr 09 2020 at 21:31):

Ah nuts, that's right... forgot about that.

view this post on Zulip Sean (Apr 09 2020 at 21:31):

the way to hide it is to either pass it in opaquely from calling code, which is ugly but can work

view this post on Zulip starseeker (Apr 09 2020 at 21:32):

<nod> - got that to work, actually...

view this post on Zulip Sean (Apr 09 2020 at 21:33):

or register it via decorator pattern as a "dm_decorator" of sorts where the dm's create it but the code is elsewhere

view this post on Zulip starseeker (Apr 09 2020 at 21:33):

I don't think the dms create the interp even now - I think the parent application passes it to them?

view this post on Zulip Sean (Apr 09 2020 at 21:33):

yes

view this post on Zulip Sean (Apr 09 2020 at 21:34):

that's how the responsbility is delegated to it

view this post on Zulip Sean (Apr 09 2020 at 21:35):

it would be something like: struct dm_window *dm_create_tk_window(interp);

view this post on Zulip starseeker (Apr 09 2020 at 21:36):

I'll see what I can do about the public API - sounds like a reasonable starting point.

view this post on Zulip starseeker (Apr 09 2020 at 21:37):

so dm_window is where Tk_Window woud hide?

view this post on Zulip Sean (Apr 09 2020 at 21:37):

we don't need to avoid Tcl_Interp being in the API entirely, but you can avoid it being in all the API portions that LIBGED calls and is aware of

view this post on Zulip Sean (Apr 09 2020 at 21:39):

struct dm_window would probably have a struct dm_window_impl* inside it that is only accessed via whatever source file had dm_create_tk_window()

view this post on Zulip Sean (Apr 09 2020 at 21:39):

that way, the Tk_Window symbol is never exposed in a public header

view this post on Zulip starseeker (Apr 09 2020 at 21:39):

Ah, gotcha.

view this post on Zulip starseeker (Apr 09 2020 at 21:40):

/me will poke at that for a bit - will probably drive me to go back and finish the license scanner :-P

view this post on Zulip Sean (Apr 09 2020 at 21:40):

As the entire entry point hinges on Tcl_Interp, even that would likely be made a dm_create_tk_window(void*interp)

view this post on Zulip Sean (Apr 09 2020 at 21:42):

so not even tcl is exposed any more. the application creating the window would know they have an interp and call it (or in the case of glfw, call a different dm_create_glfw_window(void)

view this post on Zulip starseeker (Apr 09 2020 at 21:42):

<nod>

view this post on Zulip starseeker (Apr 09 2020 at 21:45):

OK, I've got to run - thanks Sean for the help, much appreciated.

view this post on Zulip Sean (Apr 09 2020 at 21:46):

sure thing! enjoy a walk. Gorgeous day.


Last updated: Jan 09 2025 at 00:46 UTC