iOS Build Environment Help Center

Cache Unity build artifacts

append delete Kris Rok

Hi Pierre-Marie,

as it's not possible to "append" to an iOS build with recent Unity versions on Windows anymore I wonder what can be done to cache/re-use artifacts generated in a previous build, speeding up the process.

Unity wipes its output folder on build so I tried to copy the build folder to another location and restore it after Unity is done. This alone does not help, all files get recompiled (no sweet "already up to date").

Is there any way?

Sorry if this way asked (and answered) before but I could not find anything related on here.

Cheers

:: @Kris Rok added on 06 Jan ’23 · 16:13

To clarify: By "build folder" I mean the folder called "build" inside Unity's output folder.

Reply RSS

Replies

append delete #1. Pierre-Marie Baty

Hello Kris

It wouldn't help to relocate the build folder elsewhere, because the decision to recompile is based on the comparison of the source and object file's creation and modification times. As unity rewrites the source files, they get a new creation/modification time which is in all cases more recent than the last build's corresponding object files. The build system sees it and decides to recompile the file.

There is however a caching system underneath the compiler, called ccache (for compiler cache) ; if it detects that the checksum of the contents of the source file plus all of its include is identical to the checksum that was saved the last time that source file was built, the object file is not recompiled but instead pulled from the compiler cache.

It's faster than recompiling, but still, checksumming all that content with all headers included takes some time. There's no ideal solution to this problem unfortunately. What could be done here would be totally on Unity's side.

For example, they could keep some track of the checksum of the file(s) generated by the editor, and in case this checksum doesn't change compared to the last time it was created, set the file's creation and modification times to the date and time it was first created. Thus, any build system would deduce that it doesn't need to be recompiled.

append delete #2. Kris Rok

I've had some success with this workflow:

1. Using Unity's internal build method (BuildPipeLine.BuildPlayerInternal) to circumvent the "no append builds on windows" limitation :) This way, the ids inside the project.pbxproj stay the same, yay.
2. Sync this folder with another folder used for the real iOS build except (!) project.pbxproj
3. Using winmerge manually sync the changes inside project.pbxproj, skip the duplicate/clashing entries inside following nodes: buildSettings, HEADER_SEARCH_PATHS, OTHER_LDFLAGS.
4. Finally rebuild without recompilation

So besides force-enabling "append" builds the main blocker seem to be those duplicate entries. There seem to exist two kinds:

1. "Append"-related duplicates: Unity incorrectly adds entries already present _per build_. In my case:
```
HEADER_SEARCH_PATHS = (
"$(SRCROOT)/Libraries",
"$(SRCROOT)/Libraries",
"$(SRCROOT)/Libraries",
```

and

```
OTHER_LDFLAGS = (
"-lc++",
"-lc++",
"-lc++",
```

Those can easily be remedied with a post-build-check I guess.

2. Clashing flags:
```
buildSettings = {
[...]
ENABLE_BITCODE = YES;
ENABLE_BITCODE = NO;
```

I wonder what the outcome is in this case. Educated guess: NO gets used because it happens to be last?
But I don't know where they even came from and now I can't even reproduce them anymore. They disappeared when disabling/restoring your `-strip bitcode` argument. But as said, I can't reproduce it. Maybe you have some clue about it. (I'm still using version 3.53 btw!)

append delete #3. Pierre-Marie Baty

I like that kind of outsmarting stuff :) The sad thing is the thought that it could be such a simple thing to implement on Unity's side that you're working out 10x the force it would require them to achieve the same goal.

Anyway, about your observations:

These duplicate entries in HEADER_SEARCH_PATH are benign. They'll simply be passed as duplicated -I<include_path> argument to the compiler. It would certainly become a problem if you ended up passing hundreds of them, but at that point I suppose your project would have changed significantly enough for you to have triggered a total rebuild already, resetting the counters.

These duplicate entries in OTHER_LDFLAGS are benign too, they'll be passed verbatim to the linker. -lc++ simply tell the linker to link against "libc++.dylib". It won't link against the same libraries multiple times, it'll just discard the repeated arguments.

The post-build check I advise you to make is simply to remove any two identical consecutive lines in the pbxproj file. Problem solved :)

And to handle clashing flags (example 2), you could extend that logic by attempting to split each line as "key = value;", and remove any two consecutive lines that begin with an identical "key = " prefix.

As for where they come from, Unity has their own library for constructing/appending to pbxproj files. These files follow the "ASCII plist" format invented by NeXT in the 1990s. "(" denotes the start of an array value and "{" a dictionary. It simply looks like Unity's library is not able to handle duplicates in arrays and dictionaries.

Hope it helps :)

append delete #4. Kris Rok

> The post-build check I advise you to make is simply to remove any two identical consecutive lines in the pbxproj file. Problem solved :)

Haha, yeah this seemed so easy I've already implemented it right after posting :D

Now it only comes down to keeping the same timestamp of the .pbxproj as they _have_ to stay the same between builds it seems. But this just is another easy hack.

:: @Kris Rok added on 10 Jan ’23 · 11:48

Yep, now it works. Only the changed sources get recompiled.

append delete #5. Kris Rok

Ok, one last update.

The duplicates seem to be no problem (but it does not harm to remove them anyway).

So it comes down to:
1. Use a custom build script
2. Save the .pbxproj's timestamps if it exists
3. Enable an "append" build by setting `BuildOptions.AcceptExternalModificationsToPlayer`
4. Retrieve and use `BuildPipeline.BuildPlayerInternal` via reflection
5. (Optional: Clean duplicates in .pbxproj)
6. Apply the saved timestamps to the newly generated .pbxproj
7. Run the iOS build script and enjoy much faster builds :)

Reply

(Leave this as-is, it’s a trap!)

There is no need to “register”, just enter the same name + password of your choice every time.

Pro tip: Use markup to add links, quotes and more.

Moderators: Pierre-Marie Baty