digitalmars.D.learn - Explicit cast to system?
- Anonymouse (53/53) Oct 08 2022 I have some nested templated code that takes function pointers.
- tsbockman (20/29) Oct 09 2022 You might be templating more information than necessary. In your
- Anonymouse (22/29) Oct 09 2022 Yes, it was a toy example. It's complicated and I need crayons to
- user1234 (5/20) Oct 09 2022 I see what you mean, in the past I wanted `cast(pure)`,
- Anonymouse (2/4) Oct 09 2022 Darn. Okay, thanks.
I have some nested templated code that takes function pointers.
In many cases I pass it functions of identical signatures, except
some are ` safe` and others are ` system`. In those cases the
templates end up getting instantiated twice. I don't care about
the ` safe`-ness and I'd really like to just have them all
treated as ` system`, with one instantiation per unique signature.
To illustrate:
```
void foo(F)(F fun)
{
pragma(msg, F.stringof);
}
void bar() safe {}
void baz() system {}
void main()
{
foo(&bar);
foo(&baz);
}
```
Outputs:
```
void function() safe
void function()
```
I *can* do this by explicitly passing the type as a template
parameter;
```
void main()
{
foo!(void function() system)(&bar);
foo(&baz);
}
```
...but there are a lot of different signatures and I need a
general approach. There doesn't seem to be such a thing as
`cast( system)fp`.
My current solution involves some very gnarly string mixins.
```
static if (F.stringof.indexOf(" safe") != -1)
{
mixin("alias SystemF = " ~ F.stringof.replace(" safe",
" system") ~ ";");
}
else
{
alias SystemF = F;
}
```
...where `F` is `void function()`, ` safe` or ` system`. Then I
can explicitly pass `SystemF` as a compile-time parameter, and I
get my decreased instantiations.
But surely there has to be a better way?
Oct 08 2022
On Saturday, 8 October 2022 at 23:06:13 UTC, Anonymouse wrote:I have some nested templated code that takes function pointers. In many cases I pass it functions of identical signatures, except some are ` safe` and others are ` system`. In those cases the templates end up getting instantiated twice. I don't care about the ` safe`-ness and I'd really like to just have them all treated as ` system`, with one instantiation per unique signature. ... But surely there has to be a better way?You might be templating more information than necessary. In your example `foo` doesn't need to be a template at all: ```D void foo(void function() system fun) { pragma(msg, typeof(fun).stringof); } ``` If your real code needs to template the return type and parameters of `fun`, for example, consider just templating those instead of the whole function pointer type: ```D void foo(R, P...)(R function(P) system fun) { pragma(msg, typeof(fun).stringof); } ``` (Things do get awkward with `ref` and `out`, though, because D considers them to be part of the function's type rather than part of the parameter or return types. `ref` is the bane of my D meta-programming existence.)
Oct 09 2022
On Sunday, 9 October 2022 at 16:25:22 UTC, tsbockman wrote:
You might be templating more information than necessary. In
your example `foo` doesn't need to be a template at all:
```D
void foo(void function() system fun) {
pragma(msg, typeof(fun).stringof);
}
```
Yes, it was a toy example. It's complicated and I need crayons to
explain it well, but the real code is a nested function in a main
function in a template mixin, mixed into a class in a module
dedicated to it. When invoked the main function introspects
module-level functions annotated with particular UDAs and calls
them -- or doesn't, depending on other factors. Several modules
(grep says 23) then each have their own class types that mix in
this mixin, and each module's module-level functions take that
module's class as parameter. Like the hello world example does
[here](https://github.com/zorael/kameloso/blob/a471a33/source/kameloso/plugins/hello.d#L17-L25).
So the `__FUNCTION__` string of one instance of the nested
function could be (and note the ` safe`):
```
kameloso.plugins.notes.NotesPlugin.IRCPluginImpl!(Flag.no,
"kameloso.plugins.notes").onEventImpl.process!(false, false, void
function(NotesPlugin, ref const(IRCEvent)) safe).process
```
Even if I somehow manage to change the nested `process` to not be
a template I still need to instantiate the housing
`IRCPluginImpl` mixin once per class (and module), so I'm not
sure. I could just annotate everything ` system` too.
Oct 09 2022
On Saturday, 8 October 2022 at 23:06:13 UTC, Anonymouse wrote:I have some nested templated code that takes function pointers. In many cases I pass it functions of identical signatures, except some are ` safe` and others are ` system`. In those cases the templates end up getting instantiated twice. I don't care about the ` safe`-ness and I'd really like to just have them all treated as ` system`, with one instantiation per unique signature. To illustrate: [...] ...but there are a lot of different signatures and I need a general approach. There doesn't seem to be such a thing as `cast( system)fp`. My current solution involves some very gnarly string mixins.I see what you mean, in the past I wanted `cast(pure)`, `cast(nothrow)`, similarly to avoid using metaprog (or maybe was that delegates) in certain case.[...] But surely there has to be a better way?No.
Oct 09 2022
On Sunday, 9 October 2022 at 17:42:57 UTC, user1234 wrote:Darn. Okay, thanks.But surely there has to be a better way?No.
Oct 09 2022









Anonymouse <zorael gmail.com> 