The windows symbol-store; a technology so simple, yet there are nearly no sources how the store actually work. So let's get started!
There are actually 3 different ways a symbol store (and also the symbol server) can be layed out:
- level 1 store: simply stores all pdb's in a single directory, usually created when building
- level 2 store
- level 3 store
Generic terms
First we must clear up some terms or rather one term in particular: the file hash. Its used by both the level 2 and level 3 store, and it's very undocumentaded to say the least.
To retrive the hash value of an file you do:
dumpbin /HEADERS <path to exe or dll>
and look after "Debug Directories" in the output. There should be then a table like this (example for a unity executable):
Time Type Size RVA Pointer
-------- ------- -------- -------- --------
601C80A6 cv 7D 000123D0 115D0 Format: RSDS, {34D6745D-4C3F-4339-86E3-CF1FCD468701}, 1, C:\buildslave\unity\build\artifacts\WindowsPlayer\Win64_nondev_m_r\WindowsPlayer_Master_mono_x64.pdb
601C80A6 feat 14 00012450 11650 Counts: Pre-VC++ 11.00=0, C/C++=207, /GS=205, /sdl=0, guardN=205
601C80A6 coffgrp 2B8 00012464 11664
we are interested in the first entry: it tells us the original filename of the pdb as well as the hash value we are looking for! But it needs a tiny bit re-formating to make sense in a symbolstore:
{34D6745D-4C3F-4339-86E3-CF1FCD468701}, 1
becomes
34D6745D4C3F433986E3CF1FCD4687011
level 2 stores
Now to the level 2 store. A level-2 store can be recognized by the an empty "pingback.txt" file (0 bytes) and a "000Admin" folder in the root directory. It stores the pdb files in following structure:
<filename>.pdb\<hash>\<filename>.pdb
for our example above this would make
WindowsPlayer_Master_mono_x64.pdb\34D6745D4C3F433986E3CF1FCD4687011\WindowsPlayer_Master_mono_x64.pdb
and indeed, the file url on the symbolserver exists:
https://symbolserver.unity3d.com/WindowsPlayer_Master_mono_x64.pdb/34D6745D4C3F433986E3CF1FCD4687011/WindowsPlayer_Master_mono_x64.pdb
But there's more: the folder additonally contains a file named "refs.ptr". Lets open its up:
curl https://symbolserver.unity3d.com/WindowsPlayer_Master_mono_x64.pdb/34D6745D4C3F433986E3CF1FCD4687011/refs.ptr
0000000914,file,"c:\users\bvssh_~1.sai\appdata\local\temp\tmp0vo670\public\all\win64\WindowsPlayer_Master_mono_x64.pdb",pri,,Y,,
It contains a tiny bit of metadata, stored as a csv. The interesting part is the first column:
0000000914
its an index file that can be found in the "000Admin" folder. It contains references to all files related to the build of our current file.
It is also a csv file:
...
"UnityPlayer_Win64_mono_x64.pdb\A5FDEB65872C48419463B9C98EB6A1F51","c:\users\bvssh_~1.sai\appdata\local\temp\tmp0vo670\public\all\win64\UnityPlayer_Win64_mono_x64.pdb"
"WindowsPlayer_Master_mono_x64_s.pdb\34D6745D4C3F433986E3CF1FCD4687011","c:\users\bvssh_~1.sai\appdata\local\temp\tmp0vo670\public\all\win64\WindowsPlayer_Master_mono_x64_s.pdb"
"UnityPlayer_Win64_mono_x64_s.pdb\A5FDEB65872C48419463B9C98EB6A1F51","c:\users\bvssh_~1.sai\appdata\local\temp\tmp0vo670\public\all\win64\UnityPlayer_Win64_mono_x64_s.pdb"
"WindowsPlayer_Release_mono_x64.pdb\D05B9BFB5DCC40568052DF7BFA6C960F1","c:\users\bvssh_~1.sai\appdata\local\temp\tmp0vo670\public\all\win64\WindowsPlayer_Release_mono_x64.pdb"
"WindowsPlayer_Master_mono_x64.pdb\34D6745D4C3F433986E3CF1FCD4687011","c:\users\bvssh_~1.sai\appdata\local\temp\tmp0vo670\public\all\win64\WindowsPlayer_Master_mono_x64.pdb"
"WindowsPlayer_Release_mono_x64_s.pdb\D05B9BFB5DCC40568052DF7BFA6C960F1","c:\users\bvssh_~1.sai\appdata\local\temp\tmp0vo670\public\all\win64\WindowsPlayer_Release_mono_x64_s.pdb"
"UnityPlayer_UAP_x86_debug_il2cpp.pdb\D92D54AB8C2D475691CB85E308DA67C81","c:\users\bvssh_~1.sai\appdata\local\temp\tmp0vo670\public\public\win32\UnityPlayer_UAP_x86_debug_il2cpp.pdb"
"UnityPlayer_UAP_x86_master_il2cpp.pdb\6021CD859BE74AB2922EB890D9606EB41","c:\users\bvssh_~1.sai\appdata\local\temp\tmp0vo670\public\public\win32\UnityPlayer_UAP_x86_master_il2cpp.pdb"
...
As we see, there is also our file listed! Please note that the hashes in this file can apear in multiple lines, as in
...
"WindowsPlayer_Master_mono_x64_s.pdb\34D6745D4C3F433986E3CF1FCD4687011" ...
...
"WindowsPlayer_Master_mono_x64.pdb\34D6745D4C3F433986E3CF1FCD4687011" ...
...
level 3 stores
A level-3 store builds on top of a level 2 store, and can be recognized by the an empty "index2.txt" file (0 bytes) in the root directory. It stores the pdb files in following structure:
<fi>\<filename>\<hash>\<filename>.pdb>
utilitys
All the utilitys mentioned below can be obtained by downloading and installing the Windows Debugging Tools .
converting level 2 to level 3
In order to convert symbol stores, the utility "convertstore.exe" can be used:
convertstore.exe -s <StoreToConvert>
working with stores
To work with stores, one can use the "symstore.exe" utility:
Usage:
symstore add [/r] [/p] [/l] /f File /s Store /t Product [/v Version]
[/c Comment] [/d LogFile] [/compress]
symstore add [/r] [/p] [/l] [/q] /g Share /f File /x IndexFile [/a] [/d LogFile]
symstore add /y IndexFile /g Share /s Store [/p] /t Product [/v Version]
[/c Comment] [/d LogFile] [/compress]
symstore del /i ID /s Store [/d LogFile]
symstore query [/r] [/o] /f File /s Store
add Add files to server or create an index file.
del Delete a transaction from the server.
query Check if file(s) are indexed on the server.
/3 Create index2.txt when populating a new symbol server.
/f File Network path of files or directories to add.
If the named file begins with an '@' symbol, it is treated
as a response file which is expected to contain a list of
files (path and filename, 1 entry per line) to be stored.
/g Share This is the server and share where the symbol files were
originally stored. When used with /f, Share should be
identical to the beginning of the File specifier. When
used with the /y, Share should be the location of the
original symbol files, not the index file. This allows
you to later change this portion of the file path in case
you move the symbol files to a different server and share.
/i ID Transaction ID string.
/l Allows the file to be in a local directory rather than a
network path.(This option is only used with the /p option.)
/p Causes SymStore to store a pointer to the file, rather than
the file itself.
/q Don't quote fields in the index file.
/r Add files or directories recursively.
/s Store Root directory for the symbol store.
/t Product Name of the product.
/v Version Version of the product.
/c Comment Comment for the transaction.
/d LogFile Send output to LogFile instead of standard output.
/x IndexFile Causes SymStore not to store the actual symbol files in the
symbol store. Instead, information is stored which will
allow the files to be added later.
/y IndexFile This reads the data from a file created with /x.
/yi IndexFile Append a comment with the transaction ID to the end of the
index file.
/z pub | pri Pub option will only index symbols that have had the full
source information stripped. Pri will only index symbols
that contain the full source information. Both options
will index binaries.
/m <prefix> Give preference to files which have <prefix> at the beginning
of their path when storing/updating pointers.
/h pub | pri Give priority to pub or pri.
/a Causes SymStore to append new indexing information
to an existing index file. (This option is only used with
/x option.)
/o Give verbose output.
-:MSG [msg] When storing pointers, also add the provided message to the
file.ptr
-:REL Allow file.ptr paths to be relative. Implies '/l' also.
-:NOREFS Only valid during intial store creation or when used on a
store previously created with the -:NOREFS option. Omits the
creation of refs.ptr files for files and pointers stored.
Use of a store without refs.ptr precludes the ability to do
prioritization and the ability to delete transactions from the
store.
-:NOFORCECOPY
/compress When storing files, store compressed files on the server. Ignored
when storing pointers.
create a symbol-server / adding files to it
You can add files to your server by executing
symstore.exe add /s <local directory for the server files> /compress /r /f *.pdb /t <product name>
The option /compress
can be omitted; it only makes it so that compressed archives will be stored instead of the original file.To now serve the files you simply use an webserver (nginx, apache, or similar) and serve the path specified to the /s
option.
quering the store
The "symstore.exe" utility supports quering of stores, but only local stores. This means in order to query a http store you must first download it completly.
symstore.exe query /s <directory of the store> /f <file to query>