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

Refactor _d_arraysetlengthT and _d_arraysetlengthiT to a single Template-Based Implementation #21089

Closed
wants to merge 0 commits into from

Conversation

shivangshukla7020
Copy link
Contributor

@shivangshukla7020 shivangshukla7020 commented Mar 26, 2025

Description

This pull request refactors the _d_arraysetlengthT and _d_arraysetlengthiT runtime hooks to use template-based implementations instead of relying on TypeInfo. This change improves performance by enabling compile-time specialization, reducing runtime overhead, and ensuring type safety.

Key Changes

  • Implemented _d_arraysetlengthT and _d_arraysetlengthiT as templated functions.
  • Improved memory allocation and initialization using __arrayAlloc, memcpy, memmove, and emplace.
  • Ensured compatibility with mutable, immutable, and const arrays.
  • Fixed scope-related issues that caused test failures in scope.d and test23874.d.
  • Updated _d_arraycatnTX to use the new arraysetlengthT function.
  • Verified correctness through successful execution of all relevant test cases.

This update enhances runtime efficiency while maintaining compatibility with existing functionality.

Kindly review @teodutu

@dlang-bot
Copy link
Contributor

Thanks for your pull request and interest in making D better, @shivangshukla7020! We are looking forward to reviewing it, and you should be hearing from a maintainer soon.
Please verify that your PR follows this checklist:

  • My PR is fully covered with tests (you can see the coverage diff by visiting the details link of the codecov check)
  • My PR is as minimal as possible (smaller, focused PRs are easier to review than big ones)
  • I have provided a detailed rationale explaining my changes
  • New or modified functions have Ddoc comments (with Params: and Returns:)

Please see CONTRIBUTING.md for more information.


If you have addressed all reviews or aren't sure how to proceed, don't hesitate to ping us with a simple comment.

Bugzilla references

Your PR doesn't reference any Bugzilla issue.

If your PR contains non-trivial changes, please reference a Bugzilla issue or create a manual changelog.

Testing this PR locally

If you don't have a local development environment setup, you can use Digger to test this PR:

dub run digger -- build "master + dmd#21089"

@shivangshukla7020
Copy link
Contributor Author

Fixing the pre-commit checks

Copy link
Member

@teodutu teodutu left a comment

Choose a reason for hiding this comment

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

I made some initial comments. As a general note, you'll have an easier time reusing as much of the old code as you can (everything except TypeInfo) and after all tests pass you can try to make improvements. Don't reinvent the wheel too early.

Type tn = ale.e1.type.toBasetype().nextOf();
checkDefCtor(ale.loc, tn);

// Choose correct GC hook
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
// Choose correct GC hook
// Choose the correct hook

Technically only _d_arraysetlengthTTrace is a "GC hook" and all it does extra is that it prints some statistics about allocations. This one is used when compiling with the -profile=gc argument.

Comment on lines 13 to 15
* Fully templated implementation of `_d_arraysetlengthT`, removing reliance on `TypeInfo`.
* The old `_d_arraysetlengthT` and `_d_arraysetlengthiT` functions have been removed.
*/
Copy link
Member

Choose a reason for hiding this comment

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

No need to reference the old hooks.

Comment on lines 22 to 24
* This function is **now fully templated**, eliminating the need for TypeInfo-based
* `_d_arraysetlengthT` and `_d_arraysetlengthiT`, while efficiently handling both
* zero-initialized and custom-initialized types.
Copy link
Member

Choose a reason for hiding this comment

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

Again, no need to reference the old hooks. Just copy and paste the ddoc comments from rt/lifetime.d and update the parameters.

template _d_arraysetlengthTImpl(Tarr : T[], T)
/// Complete templated implementation of `_d_arraysetlengthT` and its GC profiling variant `_d_arraysetlengthTTrace`

size_t _d_arraysetlengthT(Tarr : T[], T)(
Copy link
Member

Choose a reason for hiding this comment

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

Consider updating the lowering to use this signature:

size_t _d_arraysetlengthT(Tarr : T[], T)(return scope ref Tarr arr, size_t newlength, bool isMutable)

Then the compiler should unqualify arr itself before calling the hook in expressionsem.d like so [1]. The point of this would be to only emit one template instance regardless of type qualifiers. So instead of using is (T = ...) below you'd just use isMutable). This might not work, so experiment with it first. It's best to apply this suggestion last, when everything else works.

[1]

/* Remove `inout`, `const`, `immutable` and `shared` to reduce
* the number of generated `_d_newarrayT` instances.
*/
const isShared = exp.type.nextOf.isShared();
auto t = exp.type.nextOf.unqualify(MODFlags.wild | MODFlags.const_ |
MODFlags.immutable_ | MODFlags.shared_);
tiargs.push(t);
lowering = new DotTemplateInstanceExp(exp.loc, lowering, hook, tiargs);

Copy link
Member

Choose a reason for hiding this comment

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

The diffference between _d_arraysetlenghtT and _d_arraysetlengthiT is that the former initialises all elements to 0 (so a simple memset would do it) while the latter uses the .init symbol, like you did. You should create a distinction between the two cases somewhere. Is this taken into account and I missed it?

return 0;
}

static if (is(T == immutable) || is(T == const))
Copy link
Member

Choose a reason for hiding this comment

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

Why is this important? The difference between _d_arraysetlengthT and _d_arraysetlengthiT lies in their initialisers (0 or T.init) not in mutability.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I need to update it via isMutable param where they are called right ?

Comment on lines 47 to 92
._d_arraysetlengthiT(ti, newlength, cast(void[]*)&arr);
// Allocate new array for immutable/const
auto tempArr = new U[newlength];

return arr.length;
// Copy existing elements
if (arr.ptr !is null)
{
memmove(tempArr.ptr, arr.ptr, arr.length * U.sizeof);
}

// Ensure initialization of new elements for immutable types
foreach (i; arr.length .. newlength)
{
emplace(&tempArr[i], U.init);
}

arr = cast(Tarr) tempArr;
return arr.length;
}

Copy link
Member

Choose a reason for hiding this comment

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

Don't allocate an extra array. The old hook tried to reallocate it using gc_expandArrayUsed. Keep the same inner logic.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

got it

Comment on lines 105 to 116
static size_t mulu(size_t a, size_t b, ref bool overflow)
{
size_t result = a * b;
overflow = (b != 0 && result / b != a);
return result;
}

newsize = mulu(sizeelem, newlength, overflow);
if (overflow)
{
return 0;
}
Copy link
Member

Choose a reason for hiding this comment

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

The whole point of the inline assembly code in the original hook was to verify the overflow using setc as opposed to that division. I'll guess the compiler can optimise your code to the same assembly, do these experiments once everything else passes.

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.

4 participants