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

Optimize allocations #128

Merged
merged 2 commits into from
Feb 26, 2025
Merged

Optimize allocations #128

merged 2 commits into from
Feb 26, 2025

Conversation

armanbilge
Copy link
Member

@armanbilge armanbilge commented Feb 26, 2025

  • avoids unnecessary wrapping of Runnable as () => Unit
  • replaces Scala Map and Option-based get with an ordinary js.Object

if (js.typeOf(js.Dynamic.global.setImmediate) == Undefined) {
var nextHandle = 1
val tasksByHandle = mutable.Map[Int, () => Unit]()
val tasksByHandle = new HashMap[Int, Runnable]
Copy link
Member

Choose a reason for hiding this comment

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

This is going to reach ju.HashMap for every Scala.js codebase that uses this library, potentially increasing code size. Have you considered using a plain old JS object instead?

Copy link
Member Author

Choose a reason for hiding this comment

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

@armanbilge armanbilge force-pushed the optimize-allocation branch 6 times, most recently from 5ded57b to 643145f Compare February 26, 2025 16:21
}

case None =>
val task = tasksByHandle(handle).asInstanceOf[js.UndefOr[Runnable]]
Copy link
Member

Choose a reason for hiding this comment

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

I'm pretty sure you need tasksByHandle.selectDynamic(handle) here. As is, you're calling tasksByHandle with handle as an argument.

Copy link
Member Author

Choose a reason for hiding this comment

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

Yes, you're right ... messed up the refactor from an initial try with @BracketAccess. But selectDynamic requires a string, so going to try @BracketAccess again

if (task.isDefined) {
currentlyRunningATask = true
try {
task.asInstanceOf[Runnable].run()
Copy link
Member

Choose a reason for hiding this comment

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

If you're going to cast here anyway, using js.UndefOr only to get isDefined is more trouble than it's worth. You can use js.isUndefined(task) instead.

@@ -129,7 +126,7 @@ object MacrotaskExecutor extends ExecutionContextExecutor {
val handle = nextHandle
nextHandle += 1

tasksByHandle += (handle -> k)
tasksByHandle(handle) = k.asInstanceOf[js.Any]
Copy link
Member

Choose a reason for hiding this comment

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

And here (and below as well) you'll need updateDynamic.

@armanbilge armanbilge requested a review from sjrd February 26, 2025 17:10
}

case None =>
val task = tasksByHandle(handle)
Copy link
Member

Choose a reason for hiding this comment

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

With tasksByHandle(handle) now returning a Runnable, there will be an implicit cast to Runnable here that will CCE when it's undefined.

With a custom TaskMap (which really makes sense, btw), you'll get a better mileage by having the apply return a js.UndefOr[Runnable]. Then you could actually just call foreach on it, and you won't need the cast further down. Seems like a win overall.

I wonder how we could better test that each code path actually works. The code paths that are polyfilling really old stuff are under-tested, it seems.

Copy link
Member Author

Choose a reason for hiding this comment

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

I wonder how we could better test that each code path actually works. The code paths that are polyfilling really old stuff are under-tested, it seems.

Actually, I think the reason we are not hitting the CCE in the undefined case is because it's impossible for us. I don't think we need to consider this case at all.

It was cargo-culted from the upstream, but the upstream also implements clearImmediate.
https://github.com/YuzuJS/setImmediate/blob/f1ccbfdf09cb93aadf77c4aa749ea554503b9234/setImmediate.js#L31-L33

Copy link
Member

Choose a reason for hiding this comment

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

Ah ah! That makes sense. In that case the logic looks good now. We should rename runIfPresent to run, though. Otherwise the next poor soul coming here will be confused.

@armanbilge armanbilge force-pushed the optimize-allocation branch 2 times, most recently from 5942ce0 to 1abfea2 Compare February 26, 2025 20:01
@sjrd sjrd merged commit cf9d18f into scala-js:main Feb 26, 2025
18 checks passed
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