xref: /reactos/dll/win32/shell32/CShellLink.cpp (revision b99f0b49)
1 /*
2  *
3  *      Copyright 1997  Marcus Meissner
4  *      Copyright 1998  Juergen Schmied
5  *      Copyright 2005  Mike McCormack
6  *      Copyright 2009  Andrew Hill
7  *      Copyright 2013  Dominik Hornung
8  *      Copyright 2017  Hermes Belusca-Maito
9  *
10  * This library is free software; you can redistribute it and/or
11  * modify it under the terms of the GNU Lesser General Public
12  * License as published by the Free Software Foundation; either
13  * version 2.1 of the License, or (at your option) any later version.
14  *
15  * This library is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
18  * Lesser General Public License for more details.
19  *
20  * You should have received a copy of the GNU Lesser General Public
21  * License along with this library; if not, write to the Free Software
22  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
23  *
24  * NOTES
25  *   Nearly complete information about the binary formats
26  *   of .lnk files available at http://www.wotsit.org
27  *
28  *  You can use winedump to examine the contents of a link file:
29  *   winedump lnk sc.lnk
30  *
31  *  MSI advertised shortcuts are totally undocumented.  They provide an
32  *   icon for a program that is not yet installed, and invoke MSI to
33  *   install the program when the shortcut is clicked on.  They are
34  *   created by passing a special string to SetPath, and the information
35  *   in that string is parsed an stored.
36  */
37 /*
38  * In the following is listed more documentation about the Shell Link file format,
39  * as well as its interface.
40  *
41  * General introduction about "Shell Links" (MSDN):
42  *   https://msdn.microsoft.com/en-us/library/windows/desktop/bb776891(v=vs.85).aspx
43  *
44  *
45  * Details of the file format:
46  *
47  * - Official MSDN documentation "[MS-SHLLINK]: Shell Link (.LNK) Binary File Format":
48  *   https://msdn.microsoft.com/en-us/library/dd871305.aspx
49  *
50  * - Forensics:
51  *   http://forensicswiki.org/wiki/LNK
52  *   http://computerforensics.parsonage.co.uk/downloads/TheMeaningofLIFE.pdf
53  *   https://ithreats.files.wordpress.com/2009/05/lnk_the_windows_shortcut_file_format.pdf
54  *   https://github.com/libyal/liblnk/blob/master/documentation/Windows%20Shortcut%20File%20(LNK)%20format.asciidoc
55  *
56  * - List of possible shell link header flags (SHELL_LINK_DATA_FLAGS enumeration):
57  *   https://msdn.microsoft.com/en-us/library/windows/desktop/bb762540(v=vs.85).aspx
58  *   https://msdn.microsoft.com/en-us/library/dd891314.aspx
59  *
60  *
61  * In addition to storing its target by using a PIDL, a shell link file also
62  * stores metadata to make the shell able to track the link target, in situations
63  * where the link target is moved amongst local or network directories, or moved
64  * to different volumes. For this, two structures are used:
65  *
66  * - The first and oldest one (from NewShell/WinNT4) is the "LinkInfo" structure,
67  *   stored in a serialized manner at the beginning of the shell link file:
68  *   https://msdn.microsoft.com/en-us/library/dd871404.aspx
69  *   The official API for manipulating this is located in LINKINFO.DLL .
70  *
71  * - The second, more recent one, is an extra binary block appended to the
72  *   extra-data list of the shell link file: this is the "TrackerDataBlock":
73  *   https://msdn.microsoft.com/en-us/library/dd891376.aspx
74  *   Its purpose is for link tracking, and works in coordination with the
75  *   "Distributed Link Tracking" service ('TrkWks' client, 'TrkSvr' server).
76  *   See a detailed explanation at:
77  *   http://www.serverwatch.com/tutorials/article.php/1476701/Searching-for-the-Missing-Link-Distributed-Link-Tracking.htm
78  *
79  *
80  * MSI installations most of the time create so-called "advertised shortcuts".
81  * They provide an icon for a program that may not be installed yet, and invoke
82  * MSI to install the program when the shortcut is opened (resolved).
83  * The philosophy of this approach is explained in detail inside the MSDN article
84  * "Application Resiliency: Unlock the Hidden Features of Windows Installer"
85  * (by Michael Sanford), here:
86  *   https://msdn.microsoft.com/en-us/library/aa302344.aspx
87  *
88  * This functionality is implemented by adding a binary "Darwin" data block
89  * of type "EXP_DARWIN_LINK", signature EXP_DARWIN_ID_SIG == 0xA0000006,
90  * to the shell link file:
91  *   https://msdn.microsoft.com/en-us/library/dd871369.aspx
92  * or, this could be done more simply by specifying a special link target path
93  * with the IShellLink::SetPath() function. Defining the following GUID:
94  *   SHELL32_AdvtShortcutComponent = "::{9db1186e-40df-11d1-aa8c-00c04fb67863}:"
95  * setting a target of the form:
96  *   "::{SHELL32_AdvtShortcutComponent}:<MSI_App_ID>"
97  * would automatically create the necessary binary block.
98  *
99  * With that, the target of the shortcut now becomes the MSI data. The latter
100  * is parsed from MSI and retrieved by the shell that then can run the program.
101  *
102  * This MSI functionality, dubbed "link blessing", actually originates from an
103  * older technology introduced in Internet Explorer 3 (and now obsolete since
104  * Internet Explorer 7), called "MS Internet Component Download (MSICD)", see
105  * this MSDN introductory article:
106  *   https://msdn.microsoft.com/en-us/library/aa741198(v=vs.85).aspx
107  * and leveraged in Internet Explorer 4 with "Software Update Channels", see:
108  *   https://msdn.microsoft.com/en-us/library/aa740931(v=vs.85).aspx
109  * Applications supporting this technology could present shell links having
110  * a special target, see subsection "Modifying the Shortcut" in the article:
111  *   https://msdn.microsoft.com/en-us/library/aa741201(v=vs.85).aspx#pub_shor
112  *
113  * Similarly as for the MSI shortcuts, these MSICD shortcuts are created by
114  * specifying a special link target path with the IShellLink::SetPath() function,
115  * defining the following GUID:
116  *   SHELL32_AdvtShortcutProduct = "::{9db1186f-40df-11d1-aa8c-00c04fb67863}:"
117  * and setting a target of the form:
118  *   "::{SHELL32_AdvtShortcutProduct}:<AppName>::<Path>" .
119  * A tool, called "blesslnk.exe", was also provided for automatizing the process;
120  * its ReadMe can be found in the (now outdated) MS "Internet Client SDK" (INetSDK,
121  * for MS Windows 95 and NT), whose contents can be read at:
122  *   http://www.msfn.org/board/topic/145352-new-windows-lnk-vulnerability/?page=4#comment-944223
123  * The MS INetSDK can be found at:
124  *   https://web.archive.org/web/20100924000013/http://support.microsoft.com/kb/177877
125  *
126  * Internally the shell link target of these MSICD shortcuts is converted into
127  * a binary data block of a type similar to Darwin / "EXP_DARWIN_LINK", but with
128  * a different signature EXP_LOGO3_ID_SIG == 0xA0000007 . Such shell links are
129  * called "Logo3" shortcuts. They were evoked in this user comment in "The Old
130  * New Thing" blog:
131  *   https://blogs.msdn.microsoft.com/oldnewthing/20121210-00/?p=5883#comment-1025083
132  *
133  * The shell exports the API 'SoftwareUpdateMessageBox' (in shdocvw.dll) that
134  * displays a message when an update for an application supporting this
135  * technology is available.
136  *
137  */
138 
139 #include "precomp.h"
140 
141 #include <appmgmt.h>
142 
143 WINE_DEFAULT_DEBUG_CHANNEL(shell);
144 
145 #define SHLINK_LOCAL  0
146 #define SHLINK_REMOTE 1
147 
148 /* link file formats */
149 
150 #include "pshpack1.h"
151 
152 struct LOCATION_INFO
153 {
154     DWORD        dwTotalSize;
155     DWORD        dwHeaderSize;
156     DWORD        dwFlags;
157     DWORD        dwVolTableOfs;
158     DWORD        dwLocalPathOfs;
159     DWORD        dwNetworkVolTableOfs;
160     DWORD        dwFinalPathOfs;
161 };
162 
163 struct LOCAL_VOLUME_INFO
164 {
165     DWORD        dwSize;
166     DWORD        dwType;
167     DWORD        dwVolSerial;
168     DWORD        dwVolLabelOfs;
169 };
170 
171 struct volume_info
172 {
173     DWORD        type;
174     DWORD        serial;
175     WCHAR        label[12];  /* assume 8.3 */
176 };
177 
178 #include "poppack.h"
179 
180 /* IShellLink Implementation */
181 
182 static HRESULT ShellLink_UpdatePath(LPCWSTR sPathRel, LPCWSTR path, LPCWSTR sWorkDir, LPWSTR* psPath);
183 
184 /* strdup on the process heap */
185 static LPWSTR __inline HEAP_strdupAtoW(HANDLE heap, DWORD flags, LPCSTR str)
186 {
187     INT len;
188     LPWSTR p;
189 
190     assert(str);
191 
192     len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
193     p = (LPWSTR)HeapAlloc(heap, flags, len * sizeof(WCHAR));
194     if (!p)
195         return p;
196     MultiByteToWideChar(CP_ACP, 0, str, -1, p, len);
197     return p;
198 }
199 
200 static LPWSTR __inline strdupW(LPCWSTR src)
201 {
202     LPWSTR dest;
203     if (!src) return NULL;
204     dest = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, (wcslen(src) + 1) * sizeof(WCHAR));
205     if (dest)
206         wcscpy(dest, src);
207     return dest;
208 }
209 
210 // TODO: Use it for constructor & destructor too
211 VOID CShellLink::Reset()
212 {
213     ILFree(m_pPidl);
214     m_pPidl = NULL;
215 
216     HeapFree(GetProcessHeap(), 0, m_sPath);
217     m_sPath = NULL;
218     ZeroMemory(&volume, sizeof(volume));
219 
220     HeapFree(GetProcessHeap(), 0, m_sDescription);
221     m_sDescription = NULL;
222     HeapFree(GetProcessHeap(), 0, m_sPathRel);
223     m_sPathRel = NULL;
224     HeapFree(GetProcessHeap(), 0, m_sWorkDir);
225     m_sWorkDir = NULL;
226     HeapFree(GetProcessHeap(), 0, m_sArgs);
227     m_sArgs = NULL;
228     HeapFree(GetProcessHeap(), 0, m_sIcoPath);
229     m_sIcoPath = NULL;
230 
231     m_bRunAs = FALSE;
232     m_bDirty = FALSE;
233 
234     if (m_pDBList)
235         SHFreeDataBlockList(m_pDBList);
236     m_pDBList = NULL;
237 
238     /**/sProduct = sComponent = NULL;/**/
239 }
240 
241 CShellLink::CShellLink()
242 {
243     m_Header.dwSize = sizeof(m_Header);
244     m_Header.clsid = CLSID_ShellLink;
245     m_Header.dwFlags = 0;
246 
247     m_Header.dwFileAttributes = 0;
248     ZeroMemory(&m_Header.ftCreationTime, sizeof(m_Header.ftCreationTime));
249     ZeroMemory(&m_Header.ftLastAccessTime, sizeof(m_Header.ftLastAccessTime));
250     ZeroMemory(&m_Header.ftLastWriteTime, sizeof(m_Header.ftLastWriteTime));
251     m_Header.nFileSizeLow = 0;
252 
253     m_Header.nIconIndex = 0;
254     m_Header.nShowCommand = SW_SHOWNORMAL;
255     m_Header.wHotKey = 0;
256 
257     m_pPidl = NULL;
258 
259     m_sPath = NULL;
260     ZeroMemory(&volume, sizeof(volume));
261 
262     m_sDescription = NULL;
263     m_sPathRel = NULL;
264     m_sWorkDir = NULL;
265     m_sArgs = NULL;
266     m_sIcoPath = NULL;
267     m_bRunAs = FALSE;
268     m_bDirty = FALSE;
269     m_pDBList = NULL;
270 
271     m_sLinkPath = NULL;
272     m_iIdOpen = -1;
273 
274     /**/sProduct = sComponent = NULL;/**/
275 }
276 
277 CShellLink::~CShellLink()
278 {
279     TRACE("-- destroying IShellLink(%p)\n", this);
280 
281     ILFree(m_pPidl);
282 
283     HeapFree(GetProcessHeap(), 0, m_sPath);
284 
285     HeapFree(GetProcessHeap(), 0, m_sDescription);
286     HeapFree(GetProcessHeap(), 0, m_sPathRel);
287     HeapFree(GetProcessHeap(), 0, m_sWorkDir);
288     HeapFree(GetProcessHeap(), 0, m_sArgs);
289     HeapFree(GetProcessHeap(), 0, m_sIcoPath);
290     HeapFree(GetProcessHeap(), 0, m_sLinkPath);
291     SHFreeDataBlockList(m_pDBList);
292 }
293 
294 HRESULT STDMETHODCALLTYPE CShellLink::GetClassID(CLSID *pclsid)
295 {
296     TRACE("%p %p\n", this, pclsid);
297 
298     if (pclsid == NULL)
299         return E_POINTER;
300     *pclsid = CLSID_ShellLink;
301     return S_OK;
302 }
303 
304 /************************************************************************
305  * IPersistStream_IsDirty (IPersistStream)
306  */
307 HRESULT STDMETHODCALLTYPE CShellLink::IsDirty()
308 {
309     TRACE("(%p)\n", this);
310     return (m_bDirty ? S_OK : S_FALSE);
311 }
312 
313 HRESULT STDMETHODCALLTYPE CShellLink::Load(LPCOLESTR pszFileName, DWORD dwMode)
314 {
315     TRACE("(%p, %s, %x)\n", this, debugstr_w(pszFileName), dwMode);
316 
317     if (dwMode == 0)
318         dwMode = STGM_READ | STGM_SHARE_DENY_WRITE;
319 
320     CComPtr<IStream> stm;
321     HRESULT hr = SHCreateStreamOnFileW(pszFileName, dwMode, &stm);
322     if (SUCCEEDED(hr))
323     {
324         HeapFree(GetProcessHeap(), 0, m_sLinkPath);
325         m_sLinkPath = strdupW(pszFileName);
326         hr = Load(stm);
327         ShellLink_UpdatePath(m_sPathRel, pszFileName, m_sWorkDir, &m_sPath);
328         m_bDirty = FALSE;
329     }
330     TRACE("-- returning hr %08x\n", hr);
331     return hr;
332 }
333 
334 HRESULT STDMETHODCALLTYPE CShellLink::Save(LPCOLESTR pszFileName, BOOL fRemember)
335 {
336     TRACE("(%p)->(%s)\n", this, debugstr_w(pszFileName));
337 
338     if (!pszFileName)
339         return E_FAIL;
340 
341     CComPtr<IStream> stm;
342     HRESULT hr = SHCreateStreamOnFileW(pszFileName, STGM_READWRITE | STGM_CREATE | STGM_SHARE_EXCLUSIVE, &stm);
343     if (SUCCEEDED(hr))
344     {
345         hr = Save(stm, FALSE);
346 
347         if (SUCCEEDED(hr))
348         {
349             if (m_sLinkPath)
350                 HeapFree(GetProcessHeap(), 0, m_sLinkPath);
351 
352             m_sLinkPath = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, (wcslen(pszFileName) + 1) * sizeof(WCHAR));
353             if (m_sLinkPath)
354                 wcscpy(m_sLinkPath, pszFileName);
355 
356             m_bDirty = FALSE;
357         }
358         else
359         {
360             DeleteFileW(pszFileName);
361             WARN("Failed to create shortcut %s\n", debugstr_w(pszFileName));
362         }
363     }
364 
365     return hr;
366 }
367 
368 HRESULT STDMETHODCALLTYPE CShellLink::SaveCompleted(LPCOLESTR pszFileName)
369 {
370     FIXME("(%p)->(%s)\n", this, debugstr_w(pszFileName));
371     return S_OK;
372 }
373 
374 HRESULT STDMETHODCALLTYPE CShellLink::GetCurFile(LPOLESTR *ppszFileName)
375 {
376     *ppszFileName = NULL;
377 
378     if (!m_sLinkPath)
379     {
380         /* IPersistFile::GetCurFile called before IPersistFile::Save */
381         return S_FALSE;
382     }
383 
384     *ppszFileName = (LPOLESTR)CoTaskMemAlloc((wcslen(m_sLinkPath) + 1) * sizeof(WCHAR));
385     if (!*ppszFileName)
386     {
387         /* out of memory */
388         return E_OUTOFMEMORY;
389     }
390 
391     /* copy last saved filename */
392     wcscpy(*ppszFileName, m_sLinkPath);
393 
394     return S_OK;
395 }
396 
397 static HRESULT Stream_LoadString(IStream* stm, BOOL unicode, LPWSTR *pstr)
398 {
399     TRACE("%p\n", stm);
400 
401     USHORT len;
402     DWORD count = 0;
403     HRESULT hr = stm->Read(&len, sizeof(len), &count);
404     if (FAILED(hr) || count != sizeof(len))
405         return E_FAIL;
406 
407     if (unicode)
408         len *= sizeof(WCHAR);
409 
410     TRACE("reading %d\n", len);
411     LPSTR temp = (LPSTR)HeapAlloc(GetProcessHeap(), 0, len + sizeof(WCHAR));
412     if (!temp)
413         return E_OUTOFMEMORY;
414     count = 0;
415     hr = stm->Read(temp, len, &count);
416     if (FAILED(hr) || count != len)
417     {
418         HeapFree(GetProcessHeap(), 0, temp);
419         return E_FAIL;
420     }
421 
422     TRACE("read %s\n", debugstr_an(temp, len));
423 
424     /* convert to unicode if necessary */
425     LPWSTR str;
426     if (!unicode)
427     {
428         count = MultiByteToWideChar(CP_ACP, 0, temp, len, NULL, 0);
429         str = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, (count + 1) * sizeof(WCHAR));
430         if (!str)
431         {
432             HeapFree(GetProcessHeap(), 0, temp);
433             return E_OUTOFMEMORY;
434         }
435         MultiByteToWideChar(CP_ACP, 0, temp, len, str, count);
436         HeapFree(GetProcessHeap(), 0, temp);
437     }
438     else
439     {
440         count /= sizeof(WCHAR);
441         str = (LPWSTR)temp;
442     }
443     str[count] = 0;
444 
445     *pstr = str;
446 
447     return S_OK;
448 }
449 
450 
451 /*
452  * NOTE: The following 5 functions are part of LINKINFO.DLL
453  */
454 static BOOL ShellLink_GetVolumeInfo(LPCWSTR path, CShellLink::volume_info *volume)
455 {
456     WCHAR drive[4] = { path[0], ':', '\\', 0 };
457 
458     volume->type = GetDriveTypeW(drive);
459     BOOL bRet = GetVolumeInformationW(drive, volume->label, _countof(volume->label), &volume->serial, NULL, NULL, NULL, 0);
460     TRACE("ret = %d type %d serial %08x name %s\n", bRet,
461           volume->type, volume->serial, debugstr_w(volume->label));
462     return bRet;
463 }
464 
465 static HRESULT Stream_ReadChunk(IStream* stm, LPVOID *data)
466 {
467     struct sized_chunk
468     {
469         DWORD size;
470         unsigned char data[1];
471     } *chunk;
472 
473     TRACE("%p\n", stm);
474 
475     DWORD size;
476     ULONG count;
477     HRESULT hr = stm->Read(&size, sizeof(size), &count);
478     if (FAILED(hr) || count != sizeof(size))
479         return E_FAIL;
480 
481     chunk = static_cast<sized_chunk *>(HeapAlloc(GetProcessHeap(), 0, size));
482     if (!chunk)
483         return E_OUTOFMEMORY;
484 
485     chunk->size = size;
486     hr = stm->Read(chunk->data, size - sizeof(size), &count);
487     if (FAILED(hr) || count != (size - sizeof(size)))
488     {
489         HeapFree(GetProcessHeap(), 0, chunk);
490         return E_FAIL;
491     }
492 
493     TRACE("Read %d bytes\n", chunk->size);
494 
495     *data = chunk;
496 
497     return S_OK;
498 }
499 
500 static BOOL Stream_LoadVolume(LOCAL_VOLUME_INFO *vol, CShellLink::volume_info *volume)
501 {
502     volume->serial = vol->dwVolSerial;
503     volume->type = vol->dwType;
504 
505     if (!vol->dwVolLabelOfs)
506         return FALSE;
507     if (vol->dwSize <= vol->dwVolLabelOfs)
508         return FALSE;
509     INT len = vol->dwSize - vol->dwVolLabelOfs;
510 
511     LPSTR label = (LPSTR)vol;
512     label += vol->dwVolLabelOfs;
513     MultiByteToWideChar(CP_ACP, 0, label, len, volume->label, _countof(volume->label));
514 
515     return TRUE;
516 }
517 
518 static LPWSTR Stream_LoadPath(LPCSTR p, DWORD maxlen)
519 {
520     UINT len = 0;
521 
522     while (p[len] && len < maxlen)
523         len++;
524 
525     UINT wlen = MultiByteToWideChar(CP_ACP, 0, p, len, NULL, 0);
526     LPWSTR path = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, (wlen + 1) * sizeof(WCHAR));
527     if (!path)
528         return NULL;
529     MultiByteToWideChar(CP_ACP, 0, p, len, path, wlen);
530     path[wlen] = 0;
531 
532     return path;
533 }
534 
535 static HRESULT Stream_LoadLocation(IStream *stm,
536                                    CShellLink::volume_info *volume, LPWSTR *path)
537 {
538     char *p = NULL;
539     HRESULT hr = Stream_ReadChunk(stm, (LPVOID*) &p);
540     if (FAILED(hr))
541         return hr;
542 
543     LOCATION_INFO *loc = reinterpret_cast<LOCATION_INFO *>(p);
544     if (loc->dwTotalSize < sizeof(LOCATION_INFO))
545     {
546         HeapFree(GetProcessHeap(), 0, p);
547         return E_FAIL;
548     }
549 
550     /* if there's valid local volume information, load it */
551     if (loc->dwVolTableOfs &&
552         ((loc->dwVolTableOfs + sizeof(LOCAL_VOLUME_INFO)) <= loc->dwTotalSize))
553     {
554         LOCAL_VOLUME_INFO *volume_info;
555 
556         volume_info = (LOCAL_VOLUME_INFO*) &p[loc->dwVolTableOfs];
557         Stream_LoadVolume(volume_info, volume);
558     }
559 
560     /* if there's a local path, load it */
561     DWORD n = loc->dwLocalPathOfs;
562     if (n && n < loc->dwTotalSize)
563         *path = Stream_LoadPath(&p[n], loc->dwTotalSize - n);
564 
565     TRACE("type %d serial %08x name %s path %s\n", volume->type,
566           volume->serial, debugstr_w(volume->label), debugstr_w(*path));
567 
568     HeapFree(GetProcessHeap(), 0, p);
569     return S_OK;
570 }
571 
572 
573 /*
574  * The format of the advertised shortcut info is:
575  *
576  *  Offset     Description
577  *  ------     -----------
578  *    0          Length of the block (4 bytes, usually 0x314)
579  *    4          tag (dword)
580  *    8          string data in ASCII
581  *    8+0x104    string data in UNICODE
582  *
583  * In the original Win32 implementation the buffers are not initialized
584  * to zero, so data trailing the string is random garbage.
585  */
586 HRESULT CShellLink::GetAdvertiseInfo(LPWSTR *str, DWORD dwSig)
587 {
588     LPEXP_DARWIN_LINK pInfo;
589 
590     *str = NULL;
591 
592     pInfo = (LPEXP_DARWIN_LINK)SHFindDataBlock(m_pDBList, dwSig);
593     if (!pInfo)
594         return E_FAIL;
595 
596     /* Make sure that the size of the structure is valid */
597     if (pInfo->dbh.cbSize != sizeof(*pInfo))
598     {
599         ERR("Ooops. This structure is not as expected...\n");
600         return E_FAIL;
601     }
602 
603     TRACE("dwSig %08x  string = '%s'\n", pInfo->dbh.dwSignature, debugstr_w(pInfo->szwDarwinID));
604 
605     *str = pInfo->szwDarwinID;
606     return S_OK;
607 }
608 
609 /************************************************************************
610  * IPersistStream_Load (IPersistStream)
611  */
612 HRESULT STDMETHODCALLTYPE CShellLink::Load(IStream *stm)
613 {
614     TRACE("%p %p\n", this, stm);
615 
616     if (!stm)
617         return STG_E_INVALIDPOINTER;
618 
619     /* Free all the old stuff */
620     Reset();
621 
622     ULONG dwBytesRead = 0;
623     HRESULT hr = stm->Read(&m_Header, sizeof(m_Header), &dwBytesRead);
624     if (FAILED(hr))
625         return hr;
626 
627     if (dwBytesRead != sizeof(m_Header))
628         return E_FAIL;
629     if (m_Header.dwSize != sizeof(m_Header))
630         return E_FAIL;
631     if (!IsEqualIID(m_Header.clsid, CLSID_ShellLink))
632         return E_FAIL;
633 
634     /* Load the new data in order */
635 
636     if (TRACE_ON(shell))
637     {
638         SYSTEMTIME stCreationTime;
639         SYSTEMTIME stLastAccessTime;
640         SYSTEMTIME stLastWriteTime;
641         WCHAR sTemp[MAX_PATH];
642 
643         FileTimeToSystemTime(&m_Header.ftCreationTime, &stCreationTime);
644         FileTimeToSystemTime(&m_Header.ftLastAccessTime, &stLastAccessTime);
645         FileTimeToSystemTime(&m_Header.ftLastWriteTime, &stLastWriteTime);
646 
647         GetDateFormatW(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &stCreationTime,
648                        NULL, sTemp, _countof(sTemp));
649         TRACE("-- stCreationTime: %s\n", debugstr_w(sTemp));
650         GetDateFormatW(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &stLastAccessTime,
651                        NULL, sTemp, _countof(sTemp));
652         TRACE("-- stLastAccessTime: %s\n", debugstr_w(sTemp));
653         GetDateFormatW(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &stLastWriteTime,
654                        NULL, sTemp, _countof(sTemp));
655         TRACE("-- stLastWriteTime: %s\n", debugstr_w(sTemp));
656     }
657 
658     /* load all the new stuff */
659     if (m_Header.dwFlags & SLDF_HAS_ID_LIST)
660     {
661         hr = ILLoadFromStream(stm, &m_pPidl);
662         if (FAILED(hr))
663             return hr;
664     }
665     pdump(m_pPidl);
666 
667     /* Load the location information... */
668     if (m_Header.dwFlags & SLDF_HAS_LINK_INFO)
669     {
670         hr = Stream_LoadLocation(stm, &volume, &m_sPath);
671         if (FAILED(hr))
672             return hr;
673     }
674     /* ... but if it is required not to use it, clear it */
675     if (m_Header.dwFlags & SLDF_FORCE_NO_LINKINFO)
676     {
677         HeapFree(GetProcessHeap(), 0, m_sPath);
678         m_sPath = NULL;
679         ZeroMemory(&volume, sizeof(volume));
680     }
681 
682     BOOL unicode = !!(m_Header.dwFlags & SLDF_UNICODE);
683 
684     if (m_Header.dwFlags & SLDF_HAS_NAME)
685     {
686         hr = Stream_LoadString(stm, unicode, &m_sDescription);
687         if (FAILED(hr))
688             return hr;
689         TRACE("Description  -> %s\n", debugstr_w(m_sDescription));
690     }
691 
692     if (m_Header.dwFlags & SLDF_HAS_RELPATH)
693     {
694         hr = Stream_LoadString(stm, unicode, &m_sPathRel);
695         if (FAILED(hr))
696             return hr;
697         TRACE("Relative Path-> %s\n", debugstr_w(m_sPathRel));
698     }
699 
700     if (m_Header.dwFlags & SLDF_HAS_WORKINGDIR)
701     {
702         hr = Stream_LoadString(stm, unicode, &m_sWorkDir);
703         if (FAILED(hr))
704             return hr;
705         PathRemoveBackslash(m_sWorkDir);
706         TRACE("Working Dir  -> %s\n", debugstr_w(m_sWorkDir));
707     }
708 
709     if (m_Header.dwFlags & SLDF_HAS_ARGS)
710     {
711         hr = Stream_LoadString(stm, unicode, &m_sArgs);
712         if (FAILED(hr))
713             return hr;
714         TRACE("Arguments    -> %s\n", debugstr_w(m_sArgs));
715     }
716 
717     if (m_Header.dwFlags & SLDF_HAS_ICONLOCATION)
718     {
719         hr = Stream_LoadString(stm, unicode, &m_sIcoPath);
720         if (FAILED(hr))
721             return hr;
722         TRACE("Icon file    -> %s\n", debugstr_w(m_sIcoPath));
723     }
724 
725     /* Now load the optional data block list */
726     hr = SHReadDataBlockList(stm, &m_pDBList);
727     if (FAILED(hr)) // FIXME: Should we fail?
728         return hr;
729 
730     if (TRACE_ON(shell))
731     {
732 #if (NTDDI_VERSION < NTDDI_LONGHORN)
733         if (m_Header.dwFlags & SLDF_HAS_LOGO3ID)
734         {
735             hr = GetAdvertiseInfo(&sProduct, EXP_LOGO3_ID_SIG);
736             if (SUCCEEDED(hr))
737                 TRACE("Product      -> %s\n", debugstr_w(sProduct));
738         }
739 #endif
740         if (m_Header.dwFlags & SLDF_HAS_DARWINID)
741         {
742             hr = GetAdvertiseInfo(&sComponent, EXP_DARWIN_ID_SIG);
743             if (SUCCEEDED(hr))
744                 TRACE("Component    -> %s\n", debugstr_w(sComponent));
745         }
746     }
747 
748     if (m_Header.dwFlags & SLDF_RUNAS_USER)
749         m_bRunAs = TRUE;
750     else
751         m_bRunAs = FALSE;
752 
753     TRACE("OK\n");
754 
755     pdump(m_pPidl);
756 
757     return S_OK;
758 }
759 
760 /************************************************************************
761  * Stream_WriteString
762  *
763  * Helper function for IPersistStream_Save. Writes a unicode string
764  *  with terminating nul byte to a stream, preceded by the its length.
765  */
766 static HRESULT Stream_WriteString(IStream* stm, LPCWSTR str)
767 {
768     USHORT len = wcslen(str) + 1; // FIXME: Possible overflows?
769     DWORD count;
770 
771     HRESULT hr = stm->Write(&len, sizeof(len), &count);
772     if (FAILED(hr))
773         return hr;
774 
775     len *= sizeof(WCHAR);
776 
777     hr = stm->Write(str, len, &count);
778     if (FAILED(hr))
779         return hr;
780 
781     return S_OK;
782 }
783 
784 /************************************************************************
785  * Stream_WriteLocationInfo
786  *
787  * Writes the location info to a stream
788  *
789  * FIXME: One day we might want to write the network volume information
790  *        and the final path.
791  *        Figure out how Windows deals with unicode paths here.
792  */
793 static HRESULT Stream_WriteLocationInfo(IStream* stm, LPCWSTR path,
794         CShellLink::volume_info *volume)
795 {
796     LOCAL_VOLUME_INFO *vol;
797     LOCATION_INFO *loc;
798 
799     TRACE("%p %s %p\n", stm, debugstr_w(path), volume);
800 
801     /* figure out the size of everything */
802     DWORD label_size = WideCharToMultiByte(CP_ACP, 0, volume->label, -1,
803                                       NULL, 0, NULL, NULL);
804     DWORD path_size = WideCharToMultiByte(CP_ACP, 0, path, -1,
805                                      NULL, 0, NULL, NULL);
806     DWORD volume_info_size = sizeof(*vol) + label_size;
807     DWORD final_path_size = 1;
808     DWORD total_size = sizeof(*loc) + volume_info_size + path_size + final_path_size;
809 
810     /* create pointers to everything */
811     loc = static_cast<LOCATION_INFO *>(HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, total_size));
812     vol = (LOCAL_VOLUME_INFO*) &loc[1];
813     LPSTR szLabel = (LPSTR) &vol[1];
814     LPSTR szPath = &szLabel[label_size];
815     LPSTR szFinalPath = &szPath[path_size];
816 
817     /* fill in the location information header */
818     loc->dwTotalSize = total_size;
819     loc->dwHeaderSize = sizeof(*loc);
820     loc->dwFlags = 1;
821     loc->dwVolTableOfs = sizeof(*loc);
822     loc->dwLocalPathOfs = sizeof(*loc) + volume_info_size;
823     loc->dwNetworkVolTableOfs = 0;
824     loc->dwFinalPathOfs = sizeof(*loc) + volume_info_size + path_size;
825 
826     /* fill in the volume information */
827     vol->dwSize = volume_info_size;
828     vol->dwType = volume->type;
829     vol->dwVolSerial = volume->serial;
830     vol->dwVolLabelOfs = sizeof(*vol);
831 
832     /* copy in the strings */
833     WideCharToMultiByte(CP_ACP, 0, volume->label, -1,
834                          szLabel, label_size, NULL, NULL);
835     WideCharToMultiByte(CP_ACP, 0, path, -1,
836                          szPath, path_size, NULL, NULL);
837     *szFinalPath = 0;
838 
839     ULONG count = 0;
840     HRESULT hr = stm->Write(loc, total_size, &count);
841     HeapFree(GetProcessHeap(), 0, loc);
842 
843     return hr;
844 }
845 
846 /************************************************************************
847  * IPersistStream_Save (IPersistStream)
848  *
849  * FIXME: makes assumptions about byte order
850  */
851 HRESULT STDMETHODCALLTYPE CShellLink::Save(IStream *stm, BOOL fClearDirty)
852 {
853     TRACE("%p %p %x\n", this, stm, fClearDirty);
854 
855     m_Header.dwSize = sizeof(m_Header);
856     m_Header.clsid = CLSID_ShellLink;
857 
858     /*
859      * Reset the flags: keep only the flags related to data blocks as they were
860      * already set in accordance by the different mutator member functions.
861      * The other flags will be determined now by the presence or absence of data.
862      */
863     m_Header.dwFlags &= (SLDF_RUN_WITH_SHIMLAYER | SLDF_RUNAS_USER |
864                          SLDF_RUN_IN_SEPARATE | SLDF_HAS_DARWINID |
865 #if (NTDDI_VERSION < NTDDI_LONGHORN)
866                          SLDF_HAS_LOGO3ID |
867 #endif
868                          SLDF_HAS_EXP_ICON_SZ | SLDF_HAS_EXP_SZ);
869     // TODO: When we will support Vista+ functionality, add other flags to this list.
870 
871     /* The stored strings are in UNICODE */
872     m_Header.dwFlags |= SLDF_UNICODE;
873 
874     if (m_pPidl)
875         m_Header.dwFlags |= SLDF_HAS_ID_LIST;
876     if (m_sPath)
877         m_Header.dwFlags |= SLDF_HAS_LINK_INFO;
878     if (m_sDescription && *m_sDescription)
879         m_Header.dwFlags |= SLDF_HAS_NAME;
880     if (m_sPathRel && *m_sPathRel)
881         m_Header.dwFlags |= SLDF_HAS_RELPATH;
882     if (m_sWorkDir && *m_sWorkDir)
883         m_Header.dwFlags |= SLDF_HAS_WORKINGDIR;
884     if (m_sArgs && *m_sArgs)
885         m_Header.dwFlags |= SLDF_HAS_ARGS;
886     if (m_sIcoPath && *m_sIcoPath)
887         m_Header.dwFlags |= SLDF_HAS_ICONLOCATION;
888     if (m_bRunAs)
889         m_Header.dwFlags |= SLDF_RUNAS_USER;
890 
891     /* Write the shortcut header */
892     ULONG count;
893     HRESULT hr = stm->Write(&m_Header, sizeof(m_Header), &count);
894     if (FAILED(hr))
895     {
896         ERR("Write failed\n");
897         return hr;
898     }
899 
900     /* Save the data in order */
901 
902     if (m_pPidl)
903     {
904         hr = ILSaveToStream(stm, m_pPidl);
905         if (FAILED(hr))
906         {
907             ERR("Failed to write PIDL\n");
908             return hr;
909         }
910     }
911 
912     if (m_sPath)
913     {
914         hr = Stream_WriteLocationInfo(stm, m_sPath, &volume);
915         if (FAILED(hr))
916             return hr;
917     }
918 
919     if (m_Header.dwFlags & SLDF_HAS_NAME)
920     {
921         hr = Stream_WriteString(stm, m_sDescription);
922         if (FAILED(hr))
923             return hr;
924     }
925 
926     if (m_Header.dwFlags & SLDF_HAS_RELPATH)
927     {
928         hr = Stream_WriteString(stm, m_sPathRel);
929         if (FAILED(hr))
930             return hr;
931     }
932 
933     if (m_Header.dwFlags & SLDF_HAS_WORKINGDIR)
934     {
935         hr = Stream_WriteString(stm, m_sWorkDir);
936         if (FAILED(hr))
937             return hr;
938     }
939 
940     if (m_Header.dwFlags & SLDF_HAS_ARGS)
941     {
942         hr = Stream_WriteString(stm, m_sArgs);
943         if (FAILED(hr))
944             return hr;
945     }
946 
947     if (m_Header.dwFlags & SLDF_HAS_ICONLOCATION)
948     {
949         hr = Stream_WriteString(stm, m_sIcoPath);
950         if (FAILED(hr))
951             return hr;
952     }
953 
954     /*
955      * Now save the data block list.
956      *
957      * NOTE that both advertised Product and Component are already saved
958      * inside Logo3 and Darwin data blocks in the m_pDBList list, and the
959      * m_Header.dwFlags is suitably initialized.
960      */
961     hr = SHWriteDataBlockList(stm, m_pDBList);
962     if (FAILED(hr))
963         return hr;
964 
965     /* Clear the dirty bit if requested */
966     if (fClearDirty)
967         m_bDirty = FALSE;
968 
969     return hr;
970 }
971 
972 /************************************************************************
973  * IPersistStream_GetSizeMax (IPersistStream)
974  */
975 HRESULT STDMETHODCALLTYPE CShellLink::GetSizeMax(ULARGE_INTEGER *pcbSize)
976 {
977     TRACE("(%p)\n", this);
978     return E_NOTIMPL;
979 }
980 
981 static BOOL SHELL_ExistsFileW(LPCWSTR path)
982 {
983     if (INVALID_FILE_ATTRIBUTES == GetFileAttributesW(path))
984         return FALSE;
985 
986     return TRUE;
987 }
988 
989 /**************************************************************************
990  *  ShellLink_UpdatePath
991  *    update absolute path in sPath using relative path in sPathRel
992  */
993 static HRESULT ShellLink_UpdatePath(LPCWSTR sPathRel, LPCWSTR path, LPCWSTR sWorkDir, LPWSTR* psPath)
994 {
995     if (!path || !psPath)
996         return E_INVALIDARG;
997 
998     if (!*psPath && sPathRel)
999     {
1000         WCHAR buffer[2*MAX_PATH], abs_path[2*MAX_PATH];
1001         LPWSTR final = NULL;
1002 
1003         /* first try if [directory of link file] + [relative path] finds an existing file */
1004 
1005         GetFullPathNameW(path, MAX_PATH * 2, buffer, &final);
1006         if (!final)
1007             final = buffer;
1008         wcscpy(final, sPathRel);
1009 
1010         *abs_path = '\0';
1011 
1012         if (SHELL_ExistsFileW(buffer))
1013         {
1014             if (!GetFullPathNameW(buffer, MAX_PATH, abs_path, &final))
1015                 wcscpy(abs_path, buffer);
1016         }
1017         else
1018         {
1019             /* try if [working directory] + [relative path] finds an existing file */
1020             if (sWorkDir)
1021             {
1022                 wcscpy(buffer, sWorkDir);
1023                 wcscpy(PathAddBackslashW(buffer), sPathRel);
1024 
1025                 if (SHELL_ExistsFileW(buffer))
1026                     if (!GetFullPathNameW(buffer, MAX_PATH, abs_path, &final))
1027                         wcscpy(abs_path, buffer);
1028             }
1029         }
1030 
1031         /* FIXME: This is even not enough - not all shell links can be resolved using this algorithm. */
1032         if (!*abs_path)
1033             wcscpy(abs_path, sPathRel);
1034 
1035         *psPath = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, (wcslen(abs_path) + 1) * sizeof(WCHAR));
1036         if (!*psPath)
1037             return E_OUTOFMEMORY;
1038 
1039         wcscpy(*psPath, abs_path);
1040     }
1041 
1042     return S_OK;
1043 }
1044 
1045 HRESULT STDMETHODCALLTYPE CShellLink::GetPath(LPSTR pszFile, INT cchMaxPath, WIN32_FIND_DATAA *pfd, DWORD fFlags)
1046 {
1047     HRESULT hr;
1048     LPWSTR pszFileW;
1049     WIN32_FIND_DATAW wfd;
1050 
1051     TRACE("(%p)->(pfile=%p len=%u find_data=%p flags=%u)(%s)\n",
1052           this, pszFile, cchMaxPath, pfd, fFlags, debugstr_w(m_sPath));
1053 
1054     /* Allocate a temporary UNICODE buffer */
1055     pszFileW = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, cchMaxPath * sizeof(WCHAR));
1056     if (!pszFileW)
1057         return E_OUTOFMEMORY;
1058 
1059     /* Call the UNICODE function */
1060     hr = GetPath(pszFileW, cchMaxPath, &wfd, fFlags);
1061 
1062     /* Convert the file path back to ANSI */
1063     WideCharToMultiByte(CP_ACP, 0, pszFileW, -1,
1064                         pszFile, cchMaxPath, NULL, NULL);
1065 
1066     /* Free the temporary buffer */
1067     HeapFree(GetProcessHeap(), 0, pszFileW);
1068 
1069     if (pfd)
1070     {
1071         ZeroMemory(pfd, sizeof(*pfd));
1072 
1073         /* Copy the file data if a file path was returned */
1074         if (*pszFile)
1075         {
1076             DWORD len;
1077 
1078             /* Copy the fixed part */
1079             CopyMemory(pfd, &wfd, FIELD_OFFSET(WIN32_FIND_DATAA, cFileName));
1080 
1081             /* Convert the file names to ANSI */
1082             len = lstrlenW(wfd.cFileName);
1083             WideCharToMultiByte(CP_ACP, 0, wfd.cFileName, len + 1,
1084                                 pfd->cFileName, sizeof(pfd->cFileName), NULL, NULL);
1085             len = lstrlenW(wfd.cAlternateFileName);
1086             WideCharToMultiByte(CP_ACP, 0, wfd.cAlternateFileName, len + 1,
1087                                 pfd->cAlternateFileName, sizeof(pfd->cAlternateFileName), NULL, NULL);
1088         }
1089     }
1090 
1091     return hr;
1092 }
1093 
1094 HRESULT STDMETHODCALLTYPE CShellLink::GetIDList(LPITEMIDLIST *ppidl)
1095 {
1096     TRACE("(%p)->(ppidl=%p)\n", this, ppidl);
1097 
1098     if (!m_pPidl)
1099     {
1100         *ppidl = NULL;
1101         return S_FALSE;
1102     }
1103 
1104     *ppidl = ILClone(m_pPidl);
1105     return S_OK;
1106 }
1107 
1108 HRESULT STDMETHODCALLTYPE CShellLink::SetIDList(LPCITEMIDLIST pidl)
1109 {
1110     TRACE("(%p)->(pidl=%p)\n", this, pidl);
1111     return SetTargetFromPIDLOrPath(pidl, NULL);
1112 }
1113 
1114 HRESULT STDMETHODCALLTYPE CShellLink::GetDescription(LPSTR pszName, INT cchMaxName)
1115 {
1116     TRACE("(%p)->(%p len=%u)\n", this, pszName, cchMaxName);
1117 
1118     if (cchMaxName)
1119         *pszName = 0;
1120 
1121     if (m_sDescription)
1122         WideCharToMultiByte(CP_ACP, 0, m_sDescription, -1,
1123                              pszName, cchMaxName, NULL, NULL);
1124 
1125     return S_OK;
1126 }
1127 
1128 HRESULT STDMETHODCALLTYPE CShellLink::SetDescription(LPCSTR pszName)
1129 {
1130     TRACE("(%p)->(pName=%s)\n", this, pszName);
1131 
1132     HeapFree(GetProcessHeap(), 0, m_sDescription);
1133     m_sDescription = NULL;
1134 
1135     if (pszName)
1136     {
1137         m_sDescription = HEAP_strdupAtoW(GetProcessHeap(), 0, pszName);
1138         if (!m_sDescription)
1139             return E_OUTOFMEMORY;
1140     }
1141     m_bDirty = TRUE;
1142 
1143     return S_OK;
1144 }
1145 
1146 HRESULT STDMETHODCALLTYPE CShellLink::GetWorkingDirectory(LPSTR pszDir, INT cchMaxPath)
1147 {
1148     TRACE("(%p)->(%p len=%u)\n", this, pszDir, cchMaxPath);
1149 
1150     if (cchMaxPath)
1151         *pszDir = 0;
1152 
1153     if (m_sWorkDir)
1154         WideCharToMultiByte(CP_ACP, 0, m_sWorkDir, -1,
1155                              pszDir, cchMaxPath, NULL, NULL);
1156 
1157     return S_OK;
1158 }
1159 
1160 HRESULT STDMETHODCALLTYPE CShellLink::SetWorkingDirectory(LPCSTR pszDir)
1161 {
1162     TRACE("(%p)->(dir=%s)\n", this, pszDir);
1163 
1164     HeapFree(GetProcessHeap(), 0, m_sWorkDir);
1165     m_sWorkDir = NULL;
1166 
1167     if (pszDir)
1168     {
1169         m_sWorkDir = HEAP_strdupAtoW(GetProcessHeap(), 0, pszDir);
1170         if (!m_sWorkDir)
1171             return E_OUTOFMEMORY;
1172     }
1173     m_bDirty = TRUE;
1174 
1175     return S_OK;
1176 }
1177 
1178 HRESULT STDMETHODCALLTYPE CShellLink::GetArguments(LPSTR pszArgs, INT cchMaxPath)
1179 {
1180     TRACE("(%p)->(%p len=%u)\n", this, pszArgs, cchMaxPath);
1181 
1182     if (cchMaxPath)
1183         *pszArgs = 0;
1184 
1185     if (m_sArgs)
1186         WideCharToMultiByte(CP_ACP, 0, m_sArgs, -1,
1187                              pszArgs, cchMaxPath, NULL, NULL);
1188 
1189     return S_OK;
1190 }
1191 
1192 HRESULT STDMETHODCALLTYPE CShellLink::SetArguments(LPCSTR pszArgs)
1193 {
1194     TRACE("(%p)->(args=%s)\n", this, pszArgs);
1195 
1196     HeapFree(GetProcessHeap(), 0, m_sArgs);
1197     m_sArgs = NULL;
1198 
1199     if (pszArgs)
1200     {
1201         m_sArgs = HEAP_strdupAtoW(GetProcessHeap(), 0, pszArgs);
1202         if (!m_sArgs)
1203             return E_OUTOFMEMORY;
1204     }
1205 
1206     m_bDirty = TRUE;
1207 
1208     return S_OK;
1209 }
1210 
1211 HRESULT STDMETHODCALLTYPE CShellLink::GetHotkey(WORD *pwHotkey)
1212 {
1213     TRACE("(%p)->(%p)(0x%08x)\n", this, pwHotkey, m_Header.wHotKey);
1214     *pwHotkey = m_Header.wHotKey;
1215     return S_OK;
1216 }
1217 
1218 HRESULT STDMETHODCALLTYPE CShellLink::SetHotkey(WORD wHotkey)
1219 {
1220     TRACE("(%p)->(hotkey=%x)\n", this, wHotkey);
1221 
1222     m_Header.wHotKey = wHotkey;
1223     m_bDirty = TRUE;
1224 
1225     return S_OK;
1226 }
1227 
1228 HRESULT STDMETHODCALLTYPE CShellLink::GetShowCmd(INT *piShowCmd)
1229 {
1230     TRACE("(%p)->(%p) %d\n", this, piShowCmd, m_Header.nShowCommand);
1231     *piShowCmd = m_Header.nShowCommand;
1232     return S_OK;
1233 }
1234 
1235 HRESULT STDMETHODCALLTYPE CShellLink::SetShowCmd(INT iShowCmd)
1236 {
1237     TRACE("(%p) %d\n", this, iShowCmd);
1238 
1239     m_Header.nShowCommand = iShowCmd;
1240     m_bDirty = TRUE;
1241 
1242     return S_OK;
1243 }
1244 
1245 HRESULT STDMETHODCALLTYPE CShellLink::GetIconLocation(LPSTR pszIconPath, INT cchIconPath, INT *piIcon)
1246 {
1247     HRESULT hr;
1248     LPWSTR pszIconPathW;
1249 
1250     TRACE("(%p)->(%p len=%u iicon=%p)\n", this, pszIconPath, cchIconPath, piIcon);
1251 
1252     /* Allocate a temporary UNICODE buffer */
1253     pszIconPathW = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, cchIconPath * sizeof(WCHAR));
1254     if (!pszIconPathW)
1255         return E_OUTOFMEMORY;
1256 
1257     /* Call the UNICODE function */
1258     hr = GetIconLocation(pszIconPathW, cchIconPath, piIcon);
1259 
1260     /* Convert the file path back to ANSI */
1261     WideCharToMultiByte(CP_ACP, 0, pszIconPathW, -1,
1262                         pszIconPath, cchIconPath, NULL, NULL);
1263 
1264     /* Free the temporary buffer */
1265     HeapFree(GetProcessHeap(), 0, pszIconPathW);
1266 
1267     return hr;
1268 }
1269 
1270 HRESULT STDMETHODCALLTYPE CShellLink::GetIconLocation(UINT uFlags, PSTR pszIconFile, UINT cchMax, int *piIndex, UINT *pwFlags)
1271 {
1272     HRESULT hr;
1273     LPWSTR pszIconFileW;
1274 
1275     TRACE("(%p)->(%u %p len=%u piIndex=%p pwFlags=%p)\n", this, uFlags, pszIconFile, cchMax, piIndex, pwFlags);
1276 
1277     /* Allocate a temporary UNICODE buffer */
1278     pszIconFileW = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, cchMax * sizeof(WCHAR));
1279     if (!pszIconFileW)
1280         return E_OUTOFMEMORY;
1281 
1282     /* Call the UNICODE function */
1283     hr = GetIconLocation(uFlags, pszIconFileW, cchMax, piIndex, pwFlags);
1284 
1285     /* Convert the file path back to ANSI */
1286     WideCharToMultiByte(CP_ACP, 0, pszIconFileW, -1,
1287                         pszIconFile, cchMax, NULL, NULL);
1288 
1289     /* Free the temporary buffer */
1290     HeapFree(GetProcessHeap(), 0, pszIconFileW);
1291 
1292     return hr;
1293 }
1294 
1295 HRESULT STDMETHODCALLTYPE CShellLink::Extract(PCSTR pszFile, UINT nIconIndex, HICON *phiconLarge, HICON *phiconSmall, UINT nIconSize)
1296 {
1297     TRACE("(%p)->(path=%s iicon=%u)\n", this, pszFile, nIconIndex);
1298 
1299     LPWSTR str = NULL;
1300     if (pszFile)
1301     {
1302         str = HEAP_strdupAtoW(GetProcessHeap(), 0, pszFile);
1303         if (!str)
1304             return E_OUTOFMEMORY;
1305     }
1306 
1307     HRESULT hr = Extract(str, nIconIndex, phiconLarge, phiconSmall, nIconSize);
1308 
1309     if (str)
1310         HeapFree(GetProcessHeap(), 0, str);
1311 
1312     return hr;
1313 }
1314 
1315 HRESULT STDMETHODCALLTYPE CShellLink::SetIconLocation(LPCSTR pszIconPath, INT iIcon)
1316 {
1317     TRACE("(%p)->(path=%s iicon=%u)\n", this, pszIconPath, iIcon);
1318 
1319     LPWSTR str = NULL;
1320     if (pszIconPath)
1321     {
1322         str = HEAP_strdupAtoW(GetProcessHeap(), 0, pszIconPath);
1323         if (!str)
1324             return E_OUTOFMEMORY;
1325     }
1326 
1327     HRESULT hr = SetIconLocation(str, iIcon);
1328 
1329     if (str)
1330         HeapFree(GetProcessHeap(), 0, str);
1331 
1332     return hr;
1333 }
1334 
1335 HRESULT STDMETHODCALLTYPE CShellLink::SetRelativePath(LPCSTR pszPathRel, DWORD dwReserved)
1336 {
1337     TRACE("(%p)->(path=%s %x)\n", this, pszPathRel, dwReserved);
1338 
1339     HeapFree(GetProcessHeap(), 0, m_sPathRel);
1340     m_sPathRel = NULL;
1341 
1342     if (pszPathRel)
1343     {
1344         m_sPathRel = HEAP_strdupAtoW(GetProcessHeap(), 0, pszPathRel);
1345         m_bDirty = TRUE;
1346     }
1347 
1348     return ShellLink_UpdatePath(m_sPathRel, m_sPath, m_sWorkDir, &m_sPath);
1349 }
1350 
1351 static LPWSTR
1352 shelllink_get_msi_component_path(LPWSTR component)
1353 {
1354     DWORD Result, sz = 0;
1355 
1356     Result = CommandLineFromMsiDescriptor(component, NULL, &sz);
1357     if (Result != ERROR_SUCCESS)
1358         return NULL;
1359 
1360     sz++;
1361     LPWSTR path = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, sz * sizeof(WCHAR));
1362     Result = CommandLineFromMsiDescriptor(component, path, &sz);
1363     if (Result != ERROR_SUCCESS)
1364     {
1365         HeapFree(GetProcessHeap(), 0, path);
1366         path = NULL;
1367     }
1368 
1369     TRACE("returning %s\n", debugstr_w(path));
1370 
1371     return path;
1372 }
1373 
1374 HRESULT STDMETHODCALLTYPE CShellLink::Resolve(HWND hwnd, DWORD fFlags)
1375 {
1376     HRESULT hr = S_OK;
1377     BOOL bSuccess;
1378 
1379     TRACE("(%p)->(hwnd=%p flags=%x)\n", this, hwnd, fFlags);
1380 
1381     /* FIXME: use IResolveShellLink interface? */
1382 
1383     // FIXME: See InvokeCommand().
1384 
1385 #if (NTDDI_VERSION < NTDDI_LONGHORN)
1386     // NOTE: For Logo3 (EXP_LOGO3_ID_SIG), check also for SHRestricted(REST_NOLOGO3CHANNELNOTIFY)
1387     if (m_Header.dwFlags & SLDF_HAS_LOGO3ID)
1388     {
1389         FIXME("Logo3 links are not supported yet!\n");
1390         return E_FAIL;
1391     }
1392 #endif
1393 
1394     /* Resolve Darwin (MSI) target */
1395     if (m_Header.dwFlags & SLDF_HAS_DARWINID)
1396     {
1397         LPWSTR component = NULL;
1398         hr = GetAdvertiseInfo(&component, EXP_DARWIN_ID_SIG);
1399         if (FAILED(hr))
1400             return E_FAIL;
1401 
1402         /* Clear the cached path */
1403         HeapFree(GetProcessHeap(), 0, m_sPath);
1404         m_sPath = NULL;
1405         m_sPath = shelllink_get_msi_component_path(component);
1406         if (!m_sPath)
1407             return E_FAIL;
1408     }
1409 
1410     if (!m_sPath && m_pPidl)
1411     {
1412         WCHAR buffer[MAX_PATH];
1413 
1414         bSuccess = SHGetPathFromIDListW(m_pPidl, buffer);
1415         if (bSuccess && *buffer)
1416         {
1417             m_sPath = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, (wcslen(buffer) + 1) * sizeof(WCHAR));
1418             if (!m_sPath)
1419                 return E_OUTOFMEMORY;
1420 
1421             wcscpy(m_sPath, buffer);
1422 
1423             m_bDirty = TRUE;
1424         }
1425         else
1426         {
1427             hr = S_OK;    /* don't report an error occurred while just caching information */
1428         }
1429     }
1430 
1431     // FIXME: Strange to do that here...
1432     if (!m_sIcoPath && m_sPath)
1433     {
1434         m_sIcoPath = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, (wcslen(m_sPath) + 1) * sizeof(WCHAR));
1435         if (!m_sIcoPath)
1436             return E_OUTOFMEMORY;
1437 
1438         wcscpy(m_sIcoPath, m_sPath);
1439         m_Header.nIconIndex = 0;
1440 
1441         m_bDirty = TRUE;
1442     }
1443 
1444     return hr;
1445 }
1446 
1447 HRESULT STDMETHODCALLTYPE CShellLink::SetPath(LPCSTR pszFile)
1448 {
1449     TRACE("(%p)->(path=%s)\n", this, pszFile);
1450 
1451     if (!pszFile)
1452         return E_INVALIDARG;
1453 
1454     LPWSTR str = HEAP_strdupAtoW(GetProcessHeap(), 0, pszFile);
1455     if (!str)
1456         return E_OUTOFMEMORY;
1457 
1458     HRESULT hr = SetPath(str);
1459     HeapFree(GetProcessHeap(), 0, str);
1460 
1461     return hr;
1462 }
1463 
1464 HRESULT STDMETHODCALLTYPE CShellLink::GetPath(LPWSTR pszFile, INT cchMaxPath, WIN32_FIND_DATAW *pfd, DWORD fFlags)
1465 {
1466     WCHAR buffer[MAX_PATH];
1467 
1468     TRACE("(%p)->(pfile=%p len=%u find_data=%p flags=%u)(%s)\n",
1469           this, pszFile, cchMaxPath, pfd, fFlags, debugstr_w(m_sPath));
1470 
1471     if (cchMaxPath)
1472         *pszFile = 0;
1473     // FIXME: What if cchMaxPath == 0 , or pszFile == NULL ??
1474 
1475     // FIXME: What about Darwin??
1476 
1477     /*
1478      * Retrieve the path to the target from the PIDL (if we have one).
1479      * NOTE: Do NOT use the cached path (m_sPath from link info).
1480      */
1481     if (m_pPidl && SHGetPathFromIDListW(m_pPidl, buffer))
1482     {
1483         if (fFlags & SLGP_SHORTPATH)
1484             GetShortPathNameW(buffer, buffer, _countof(buffer));
1485         // FIXME: Add support for SLGP_UNCPRIORITY
1486     }
1487     else
1488     {
1489         *buffer = 0;
1490     }
1491 
1492     /* If we have a FindData structure, initialize it */
1493     if (pfd)
1494     {
1495         ZeroMemory(pfd, sizeof(*pfd));
1496 
1497         /* Copy the file data if the target is a file path */
1498         if (*buffer)
1499         {
1500             pfd->dwFileAttributes = m_Header.dwFileAttributes;
1501             pfd->ftCreationTime   = m_Header.ftCreationTime;
1502             pfd->ftLastAccessTime = m_Header.ftLastAccessTime;
1503             pfd->ftLastWriteTime  = m_Header.ftLastWriteTime;
1504             pfd->nFileSizeHigh    = 0;
1505             pfd->nFileSizeLow     = m_Header.nFileSizeLow;
1506 
1507             /*
1508              * Build temporarily a short path in pfd->cFileName (of size MAX_PATH),
1509              * then extract and store the short file name in pfd->cAlternateFileName.
1510              */
1511             GetShortPathNameW(buffer, pfd->cFileName, _countof(pfd->cFileName));
1512             lstrcpynW(pfd->cAlternateFileName,
1513                       PathFindFileNameW(pfd->cFileName),
1514                       _countof(pfd->cAlternateFileName));
1515 
1516             /* Now extract and store the long file name in pfd->cFileName */
1517             lstrcpynW(pfd->cFileName,
1518                       PathFindFileNameW(buffer),
1519                       _countof(pfd->cFileName));
1520         }
1521     }
1522 
1523     /* Finally check if we have a raw path the user actually wants to retrieve */
1524     if ((fFlags & SLGP_RAWPATH) && (m_Header.dwFlags & SLDF_HAS_EXP_SZ))
1525     {
1526         /* Search for a target environment block */
1527         LPEXP_SZ_LINK pInfo;
1528         pInfo = (LPEXP_SZ_LINK)SHFindDataBlock(m_pDBList, EXP_SZ_LINK_SIG);
1529         if (pInfo && (pInfo->cbSize == sizeof(*pInfo)))
1530             lstrcpynW(buffer, pInfo->szwTarget, cchMaxPath);
1531     }
1532 
1533     /* For diagnostics purposes only... */
1534     // NOTE: SLGP_UNCPRIORITY is unsupported
1535     fFlags &= ~(SLGP_RAWPATH | SLGP_SHORTPATH);
1536     if (fFlags) FIXME("(%p): Unsupported flags %lu\n", this, fFlags);
1537 
1538     /* Copy the data back to the user */
1539     if (*buffer)
1540         lstrcpynW(pszFile, buffer, cchMaxPath);
1541 
1542     return (*buffer ? S_OK : S_FALSE);
1543 }
1544 
1545 HRESULT STDMETHODCALLTYPE CShellLink::GetDescription(LPWSTR pszName, INT cchMaxName)
1546 {
1547     TRACE("(%p)->(%p len=%u)\n", this, pszName, cchMaxName);
1548 
1549     *pszName = 0;
1550     if (m_sDescription)
1551         lstrcpynW(pszName, m_sDescription, cchMaxName);
1552 
1553     return S_OK;
1554 }
1555 
1556 HRESULT STDMETHODCALLTYPE CShellLink::SetDescription(LPCWSTR pszName)
1557 {
1558     TRACE("(%p)->(desc=%s)\n", this, debugstr_w(pszName));
1559 
1560     HeapFree(GetProcessHeap(), 0, m_sDescription);
1561     if (pszName)
1562     {
1563         m_sDescription = (LPWSTR)HeapAlloc(GetProcessHeap(), 0,
1564                                          (wcslen(pszName) + 1) * sizeof(WCHAR));
1565         if (!m_sDescription)
1566             return E_OUTOFMEMORY;
1567 
1568         wcscpy(m_sDescription, pszName);
1569     }
1570     else
1571         m_sDescription = NULL;
1572 
1573     m_bDirty = TRUE;
1574 
1575     return S_OK;
1576 }
1577 
1578 HRESULT STDMETHODCALLTYPE CShellLink::GetWorkingDirectory(LPWSTR pszDir, INT cchMaxPath)
1579 {
1580     TRACE("(%p)->(%p len %u)\n", this, pszDir, cchMaxPath);
1581 
1582     if (cchMaxPath)
1583         *pszDir = 0;
1584 
1585     if (m_sWorkDir)
1586         lstrcpynW(pszDir, m_sWorkDir, cchMaxPath);
1587 
1588     return S_OK;
1589 }
1590 
1591 HRESULT STDMETHODCALLTYPE CShellLink::SetWorkingDirectory(LPCWSTR pszDir)
1592 {
1593     TRACE("(%p)->(dir=%s)\n", this, debugstr_w(pszDir));
1594 
1595     HeapFree(GetProcessHeap(), 0, m_sWorkDir);
1596     if (pszDir)
1597     {
1598         m_sWorkDir = (LPWSTR)HeapAlloc(GetProcessHeap(), 0,
1599                                      (wcslen(pszDir) + 1) * sizeof(WCHAR));
1600         if (!m_sWorkDir)
1601             return E_OUTOFMEMORY;
1602         wcscpy(m_sWorkDir, pszDir);
1603     }
1604     else
1605         m_sWorkDir = NULL;
1606 
1607     m_bDirty = TRUE;
1608 
1609     return S_OK;
1610 }
1611 
1612 HRESULT STDMETHODCALLTYPE CShellLink::GetArguments(LPWSTR pszArgs, INT cchMaxPath)
1613 {
1614     TRACE("(%p)->(%p len=%u)\n", this, pszArgs, cchMaxPath);
1615 
1616     if (cchMaxPath)
1617         *pszArgs = 0;
1618 
1619     if (m_sArgs)
1620         lstrcpynW(pszArgs, m_sArgs, cchMaxPath);
1621 
1622     return S_OK;
1623 }
1624 
1625 HRESULT STDMETHODCALLTYPE CShellLink::SetArguments(LPCWSTR pszArgs)
1626 {
1627     TRACE("(%p)->(args=%s)\n", this, debugstr_w(pszArgs));
1628 
1629     HeapFree(GetProcessHeap(), 0, m_sArgs);
1630     if (pszArgs)
1631     {
1632         m_sArgs = (LPWSTR)HeapAlloc(GetProcessHeap(), 0,
1633                                   (wcslen(pszArgs) + 1) * sizeof(WCHAR));
1634         if (!m_sArgs)
1635             return E_OUTOFMEMORY;
1636 
1637         wcscpy(m_sArgs, pszArgs);
1638     }
1639     else
1640         m_sArgs = NULL;
1641 
1642     m_bDirty = TRUE;
1643 
1644     return S_OK;
1645 }
1646 
1647 HRESULT STDMETHODCALLTYPE CShellLink::GetIconLocation(LPWSTR pszIconPath, INT cchIconPath, INT *piIcon)
1648 {
1649     TRACE("(%p)->(%p len=%u iicon=%p)\n", this, pszIconPath, cchIconPath, piIcon);
1650 
1651     if (cchIconPath)
1652         *pszIconPath = 0;
1653 
1654     *piIcon = 0;
1655 
1656     /* Update the original icon path location */
1657     if (m_Header.dwFlags & SLDF_HAS_EXP_ICON_SZ)
1658     {
1659         WCHAR szPath[MAX_PATH];
1660 
1661         /* Search for an icon environment block */
1662         LPEXP_SZ_LINK pInfo;
1663         pInfo = (LPEXP_SZ_LINK)SHFindDataBlock(m_pDBList, EXP_SZ_ICON_SIG);
1664         if (pInfo && (pInfo->cbSize == sizeof(*pInfo)))
1665         {
1666             m_Header.dwFlags &= ~SLDF_HAS_ICONLOCATION;
1667             HeapFree(GetProcessHeap(), 0, m_sIcoPath);
1668             m_sIcoPath = NULL;
1669 
1670             SHExpandEnvironmentStringsW(pInfo->szwTarget, szPath, _countof(szPath));
1671 
1672             m_sIcoPath = (LPWSTR)HeapAlloc(GetProcessHeap(), 0,
1673                                            (wcslen(szPath) + 1) * sizeof(WCHAR));
1674             if (!m_sIcoPath)
1675                 return E_OUTOFMEMORY;
1676 
1677             wcscpy(m_sIcoPath, szPath);
1678             m_Header.dwFlags |= SLDF_HAS_ICONLOCATION;
1679 
1680             m_bDirty = TRUE;
1681         }
1682     }
1683 
1684     *piIcon = m_Header.nIconIndex;
1685 
1686     if (m_sIcoPath)
1687         lstrcpynW(pszIconPath, m_sIcoPath, cchIconPath);
1688 
1689     return S_OK;
1690 }
1691 
1692 static HRESULT SHELL_PidlGetIconLocationW(PCIDLIST_ABSOLUTE pidl,
1693         UINT uFlags, PWSTR pszIconFile, UINT cchMax, int *piIndex, UINT *pwFlags)
1694 {
1695     LPCITEMIDLIST pidlLast;
1696     CComPtr<IShellFolder> psf;
1697 
1698     HRESULT hr = SHBindToParent(pidl, IID_PPV_ARG(IShellFolder, &psf), &pidlLast);
1699     if (FAILED_UNEXPECTEDLY(hr))
1700         return hr;
1701 
1702     CComPtr<IExtractIconW> pei;
1703     hr = psf->GetUIObjectOf(0, 1, &pidlLast, IID_NULL_PPV_ARG(IExtractIconW, &pei));
1704     if (FAILED_UNEXPECTEDLY(hr))
1705         return hr;
1706 
1707     hr = pei->GetIconLocation(uFlags, pszIconFile, cchMax, piIndex, pwFlags);
1708     if (FAILED_UNEXPECTEDLY(hr))
1709         return hr;
1710 
1711     return S_OK;
1712 }
1713 
1714 HRESULT STDMETHODCALLTYPE CShellLink::GetIconLocation(UINT uFlags, PWSTR pszIconFile, UINT cchMax, int *piIndex, UINT *pwFlags)
1715 {
1716     HRESULT hr;
1717     /*
1718      * It is possible for a shell link to point to another shell link,
1719      * and in particular there is the possibility to point to itself.
1720      * Now, suppose we ask such a link to retrieve its associated icon.
1721      * This function would be called, and due to COM would be called again
1722      * recursively. To solve this issue, we forbid calling GetIconLocation()
1723      * with GIL_FORSHORTCUT set in uFlags, as done by Windows (shown by tests).
1724      */
1725     if (uFlags & GIL_FORSHORTCUT)
1726         return E_INVALIDARG;
1727 
1728     /*
1729      * Now, we set GIL_FORSHORTCUT so that: i) we allow the icon extractor
1730      * of the target to give us a suited icon, and ii) we protect ourselves
1731      * against recursive call.
1732      */
1733     uFlags |= GIL_FORSHORTCUT;
1734 
1735     if (m_pPidl || m_sPath)
1736     {
1737         /* first look for an icon using the PIDL (if present) */
1738         if (m_pPidl)
1739             hr = SHELL_PidlGetIconLocationW(m_pPidl, uFlags, pszIconFile, cchMax, piIndex, pwFlags);
1740         else
1741             hr = E_FAIL;
1742 
1743 #if 0 // FIXME: Analyse further whether this is needed...
1744         /* if we couldn't find an icon yet, look for it using the file system path */
1745         if (FAILED(hr) && m_sPath)
1746         {
1747             LPITEMIDLIST pidl;
1748             CComPtr<IShellFolder> pdsk;
1749 
1750             hr = SHGetDesktopFolder(&pdsk);
1751 
1752             /* LPITEMIDLIST pidl = ILCreateFromPathW(sPath); */
1753             hr = pdsk->ParseDisplayName(0, NULL, m_sPath, NULL, &pidl, NULL);
1754             if (SUCCEEDED(hr))
1755             {
1756                 hr = SHELL_PidlGetIconLocationW(pidl, uFlags, pszIconFile, cchMax, piIndex, pwFlags);
1757                 SHFree(pidl);
1758             }
1759         }
1760 #endif
1761         return hr;
1762     }
1763 
1764     return S_OK;
1765 }
1766 
1767 HRESULT STDMETHODCALLTYPE CShellLink::Extract(PCWSTR pszFile, UINT nIconIndex, HICON *phiconLarge, HICON *phiconSmall, UINT nIconSize)
1768 {
1769     UNIMPLEMENTED;
1770     return E_FAIL;
1771 }
1772 
1773 #if 0
1774 /* Extends the functionality of PathUnExpandEnvStringsW */
1775 BOOL PathFullyUnExpandEnvStringsW(
1776     _In_  LPCWSTR pszPath,
1777     _Out_ LPWSTR  pszBuf,
1778     _In_  UINT    cchBuf)
1779 {
1780     BOOL Ret = FALSE; // Set to TRUE as soon as PathUnExpandEnvStrings starts unexpanding.
1781     BOOL res;
1782     LPCWSTR p;
1783 
1784     // *pszBuf = L'\0';
1785     while (*pszPath && cchBuf > 0)
1786     {
1787         /* Attempt unexpanding the path */
1788         res = PathUnExpandEnvStringsW(pszPath, pszBuf, cchBuf);
1789         if (!res)
1790         {
1791             /* The unexpansion failed. Try to find a path delimiter. */
1792             p = wcspbrk(pszPath, L" /\\:*?\"<>|%");
1793             if (!p) /* None found, we will copy the remaining path */
1794                 p = pszPath + wcslen(pszPath);
1795             else    /* Found one, we will copy the delimiter and skip it */
1796                 ++p;
1797             /* If we overflow, we cannot unexpand more, so return FALSE */
1798             if (p - pszPath >= cchBuf)
1799                 return FALSE; // *pszBuf = L'\0';
1800 
1801             /* Copy the untouched portion of path up to the delimiter, included */
1802             wcsncpy(pszBuf, pszPath, p - pszPath);
1803             pszBuf[p - pszPath] = L'\0'; // NULL-terminate
1804 
1805             /* Advance the pointers and decrease the remaining buffer size */
1806             cchBuf -= (p - pszPath);
1807             pszBuf += (p - pszPath);
1808             pszPath += (p - pszPath);
1809         }
1810         else
1811         {
1812             /*
1813              * The unexpansion succeeded. Skip the unexpanded part by trying
1814              * to find where the original path and the unexpanded string
1815              * become different.
1816              * NOTE: An alternative(?) would be to stop also at the last
1817              * path delimiter encountered in the loop (i.e. would be the
1818              * first path delimiter in the strings).
1819              */
1820             LPWSTR q;
1821 
1822             /*
1823              * The algorithm starts at the end of the strings and loops back
1824              * while the characters are equal, until it finds a discrepancy.
1825              */
1826             p = pszPath + wcslen(pszPath);
1827             q = pszBuf + wcslen(pszBuf); // This wcslen should be < cchBuf
1828             while ((*p == *q) && (p > pszPath) && (q > pszBuf))
1829             {
1830                 --p; --q;
1831             }
1832             /* Skip discrepancy */
1833             ++p; ++q;
1834 
1835             /* Advance the pointers and decrease the remaining buffer size */
1836             cchBuf -= (q - pszBuf);
1837             pszBuf = q;
1838             pszPath = p;
1839 
1840             Ret = TRUE;
1841         }
1842     }
1843 
1844     return Ret;
1845 }
1846 #endif
1847 
1848 HRESULT STDMETHODCALLTYPE CShellLink::SetIconLocation(LPCWSTR pszIconPath, INT iIcon)
1849 {
1850     HRESULT hr = E_FAIL;
1851     WCHAR szUnExpIconPath[MAX_PATH];
1852 
1853     TRACE("(%p)->(path=%s iicon=%u)\n", this, debugstr_w(pszIconPath), iIcon);
1854 
1855     if (pszIconPath)
1856     {
1857         /* Try to fully unexpand the icon path */
1858 
1859         /*
1860          * Check whether the user-given file path contains unexpanded
1861          * environment variables. If so, create a target environment block.
1862          * Note that in this block we will store the user-given path.
1863          * It will contain the unexpanded environment variables, but
1864          * it can also contain already expanded path that the user does
1865          * not want to see them unexpanded (e.g. so that they always
1866          * refer to the same place even if the would-be corresponding
1867          * environment variable could change).
1868          */
1869         // FIXME: http://stackoverflow.com/questions/2976489/ishelllinkseticonlocation-translates-my-icon-path-into-program-files-which-i
1870         // if (PathFullyUnExpandEnvStringsW(pszIconPath, szUnExpIconPath, _countof(szUnExpIconPath)))
1871         SHExpandEnvironmentStringsW(pszIconPath, szUnExpIconPath, _countof(szUnExpIconPath));
1872         if (wcscmp(pszIconPath, szUnExpIconPath) != 0)
1873         {
1874             /* Unexpansion succeeded, so we need an icon environment block */
1875             EXP_SZ_LINK buffer;
1876             LPEXP_SZ_LINK pInfo;
1877 
1878             pInfo = (LPEXP_SZ_LINK)SHFindDataBlock(m_pDBList, EXP_SZ_ICON_SIG);
1879             if (pInfo)
1880             {
1881                 /* Make sure that the size of the structure is valid */
1882                 if (pInfo->cbSize != sizeof(*pInfo))
1883                 {
1884                     ERR("Ooops. This structure is not as expected...\n");
1885 
1886                     /* Invalid structure, remove it altogether */
1887                     m_Header.dwFlags &= ~SLDF_HAS_EXP_ICON_SZ;
1888                     RemoveDataBlock(EXP_SZ_ICON_SIG);
1889 
1890                     /* Reset the pointer and go use the static buffer */
1891                     pInfo = NULL;
1892                 }
1893             }
1894             if (!pInfo)
1895             {
1896                 /* Use the static buffer */
1897                 pInfo = &buffer;
1898                 buffer.cbSize = sizeof(buffer);
1899                 buffer.dwSignature = EXP_SZ_ICON_SIG;
1900             }
1901 
1902             lstrcpynW(pInfo->szwTarget, szUnExpIconPath, _countof(pInfo->szwTarget));
1903             WideCharToMultiByte(CP_ACP, 0, szUnExpIconPath, -1,
1904                                 pInfo->szTarget, _countof(pInfo->szTarget), NULL, NULL);
1905 
1906             hr = S_OK;
1907             if (pInfo == &buffer)
1908                 hr = AddDataBlock(pInfo);
1909             if (hr == S_OK)
1910                 m_Header.dwFlags |= SLDF_HAS_EXP_ICON_SZ;
1911         }
1912         else
1913         {
1914             /* Unexpansion failed, so we need to remove any icon environment block */
1915             m_Header.dwFlags &= ~SLDF_HAS_EXP_ICON_SZ;
1916             RemoveDataBlock(EXP_SZ_ICON_SIG);
1917         }
1918     }
1919 
1920     /* Store the original icon path location (this one may contain unexpanded environment strings) */
1921     if (pszIconPath)
1922     {
1923         m_Header.dwFlags &= ~SLDF_HAS_ICONLOCATION;
1924         HeapFree(GetProcessHeap(), 0, m_sIcoPath);
1925         m_sIcoPath = NULL;
1926 
1927         m_sIcoPath = (LPWSTR)HeapAlloc(GetProcessHeap(), 0,
1928                                      (wcslen(pszIconPath) + 1) * sizeof(WCHAR));
1929         if (!m_sIcoPath)
1930             return E_OUTOFMEMORY;
1931 
1932         wcscpy(m_sIcoPath, pszIconPath);
1933         m_Header.dwFlags |= SLDF_HAS_ICONLOCATION;
1934     }
1935 
1936     hr = S_OK;
1937 
1938     m_Header.nIconIndex = iIcon;
1939     m_bDirty = TRUE;
1940 
1941     return hr;
1942 }
1943 
1944 HRESULT STDMETHODCALLTYPE CShellLink::SetRelativePath(LPCWSTR pszPathRel, DWORD dwReserved)
1945 {
1946     TRACE("(%p)->(path=%s %x)\n", this, debugstr_w(pszPathRel), dwReserved);
1947 
1948     HeapFree(GetProcessHeap(), 0, m_sPathRel);
1949     if (pszPathRel)
1950     {
1951         m_sPathRel = (LPWSTR)HeapAlloc(GetProcessHeap(), 0,
1952                                       (wcslen(pszPathRel) + 1) * sizeof(WCHAR));
1953         if (!m_sPathRel)
1954             return E_OUTOFMEMORY;
1955         wcscpy(m_sPathRel, pszPathRel);
1956     }
1957     else
1958         m_sPathRel = NULL;
1959 
1960     m_bDirty = TRUE;
1961 
1962     return ShellLink_UpdatePath(m_sPathRel, m_sPath, m_sWorkDir, &m_sPath);
1963 }
1964 
1965 static LPWSTR GetAdvertisedArg(LPCWSTR str)
1966 {
1967     if (!str)
1968         return NULL;
1969 
1970     LPCWSTR p = wcschr(str, L':');
1971     if (!p)
1972         return NULL;
1973 
1974     DWORD len = p - str;
1975     LPWSTR ret = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, sizeof(WCHAR) * (len + 1));
1976     if (!ret)
1977         return ret;
1978 
1979     memcpy(ret, str, sizeof(WCHAR)*len);
1980     ret[len] = 0;
1981     return ret;
1982 }
1983 
1984 HRESULT CShellLink::WriteAdvertiseInfo(LPCWSTR string, DWORD dwSig)
1985 {
1986     EXP_DARWIN_LINK buffer;
1987     LPEXP_DARWIN_LINK pInfo;
1988 
1989     if (   (dwSig != EXP_DARWIN_ID_SIG)
1990 #if (NTDDI_VERSION < NTDDI_LONGHORN)
1991         && (dwSig != EXP_LOGO3_ID_SIG)
1992 #endif
1993         )
1994     {
1995         return E_INVALIDARG;
1996     }
1997 
1998     if (!string)
1999         return S_FALSE;
2000 
2001     pInfo = (LPEXP_DARWIN_LINK)SHFindDataBlock(m_pDBList, dwSig);
2002     if (pInfo)
2003     {
2004         /* Make sure that the size of the structure is valid */
2005         if (pInfo->dbh.cbSize != sizeof(*pInfo))
2006         {
2007             ERR("Ooops. This structure is not as expected...\n");
2008 
2009             /* Invalid structure, remove it altogether */
2010             if (dwSig == EXP_DARWIN_ID_SIG)
2011                 m_Header.dwFlags &= ~SLDF_HAS_DARWINID;
2012 #if (NTDDI_VERSION < NTDDI_LONGHORN)
2013             else if (dwSig == EXP_LOGO3_ID_SIG)
2014                 m_Header.dwFlags &= ~SLDF_HAS_LOGO3ID;
2015 #endif
2016             RemoveDataBlock(dwSig);
2017 
2018             /* Reset the pointer and go use the static buffer */
2019             pInfo = NULL;
2020         }
2021     }
2022     if (!pInfo)
2023     {
2024         /* Use the static buffer */
2025         pInfo = &buffer;
2026         buffer.dbh.cbSize = sizeof(buffer);
2027         buffer.dbh.dwSignature = dwSig;
2028     }
2029 
2030     lstrcpynW(pInfo->szwDarwinID, string, _countof(pInfo->szwDarwinID));
2031     WideCharToMultiByte(CP_ACP, 0, string, -1,
2032                         pInfo->szDarwinID, _countof(pInfo->szDarwinID), NULL, NULL);
2033 
2034     HRESULT hr = S_OK;
2035     if (pInfo == &buffer)
2036         hr = AddDataBlock(pInfo);
2037     if (hr == S_OK)
2038     {
2039         if (dwSig == EXP_DARWIN_ID_SIG)
2040             m_Header.dwFlags |= SLDF_HAS_DARWINID;
2041 #if (NTDDI_VERSION < NTDDI_LONGHORN)
2042         else if (dwSig == EXP_LOGO3_ID_SIG)
2043             m_Header.dwFlags |= SLDF_HAS_LOGO3ID;
2044 #endif
2045     }
2046 
2047     return hr;
2048 }
2049 
2050 HRESULT CShellLink::SetAdvertiseInfo(LPCWSTR str)
2051 {
2052     HRESULT hr;
2053     LPCWSTR szComponent = NULL, szProduct = NULL, p;
2054     INT len;
2055     GUID guid;
2056     WCHAR szGuid[38+1];
2057 
2058     /**/sProduct = sComponent = NULL;/**/
2059 
2060     while (str[0])
2061     {
2062         /* each segment must start with two colons */
2063         if (str[0] != ':' || str[1] != ':')
2064             return E_FAIL;
2065 
2066         /* the last segment is just two colons */
2067         if (!str[2])
2068             break;
2069         str += 2;
2070 
2071         /* there must be a colon straight after a guid */
2072         p = wcschr(str, L':');
2073         if (!p)
2074             return E_FAIL;
2075         len = p - str;
2076         if (len != 38)
2077             return E_FAIL;
2078 
2079         /* get the guid, and check if it's validly formatted */
2080         memcpy(szGuid, str, sizeof(WCHAR)*len);
2081         szGuid[len] = 0;
2082 
2083         hr = CLSIDFromString(szGuid, &guid);
2084         if (hr != S_OK)
2085             return hr;
2086         str = p + 1;
2087 
2088         /* match it up to a guid that we care about */
2089         if (IsEqualGUID(guid, SHELL32_AdvtShortcutComponent) && !szComponent)
2090             szComponent = str; /* Darwin */
2091         else if (IsEqualGUID(guid, SHELL32_AdvtShortcutProduct) && !szProduct)
2092             szProduct = str;   /* Logo3  */
2093         else
2094             return E_FAIL;
2095 
2096         /* skip to the next field */
2097         str = wcschr(str, L':');
2098         if (!str)
2099             return E_FAIL;
2100     }
2101 
2102     /* we have to have a component for an advertised shortcut */
2103     if (!szComponent)
2104         return E_FAIL;
2105 
2106     szComponent = GetAdvertisedArg(szComponent);
2107     szProduct = GetAdvertisedArg(szProduct);
2108 
2109     hr = WriteAdvertiseInfo(szComponent, EXP_DARWIN_ID_SIG);
2110     // if (FAILED(hr))
2111         // return hr;
2112 #if (NTDDI_VERSION < NTDDI_LONGHORN)
2113     hr = WriteAdvertiseInfo(szProduct, EXP_LOGO3_ID_SIG);
2114     // if (FAILED(hr))
2115         // return hr;
2116 #endif
2117 
2118     HeapFree(GetProcessHeap(), 0, (PVOID)szComponent);
2119     HeapFree(GetProcessHeap(), 0, (PVOID)szProduct);
2120 
2121     if (TRACE_ON(shell))
2122     {
2123         GetAdvertiseInfo(&sComponent, EXP_DARWIN_ID_SIG);
2124         TRACE("Component = %s\n", debugstr_w(sComponent));
2125 #if (NTDDI_VERSION < NTDDI_LONGHORN)
2126         GetAdvertiseInfo(&sProduct, EXP_LOGO3_ID_SIG);
2127         TRACE("Product = %s\n", debugstr_w(sProduct));
2128 #endif
2129     }
2130 
2131     return S_OK;
2132 }
2133 
2134 /*
2135  * Since the real PathResolve (from Wine) is unimplemented at the moment,
2136  * we use this local implementation, until a better one is written (using
2137  * code parts of the SHELL_xxx helpers in Wine's shellpath.c).
2138  */
2139 static BOOL HACKISH_PathResolve(
2140     IN OUT PWSTR pszPath,
2141     IN PZPCWSTR dirs OPTIONAL,
2142     IN UINT fFlags)
2143 {
2144     // FIXME: This is unimplemented!!!
2145 #if 0
2146     return PathResolve(pszPath, dirs, fFlags);
2147 #else
2148     BOOL Success = FALSE;
2149     USHORT i;
2150     LPWSTR fname = NULL;
2151     WCHAR szPath[MAX_PATH];
2152 
2153     /* First, search for a valid existing path */
2154 
2155     // NOTE: See also: SHELL_FindExecutable()
2156 
2157     /*
2158      * List of extensions searched for, by PathResolve with the flag
2159      * PRF_TRYPROGRAMEXTENSIONS == PRF_EXECUTABLE | PRF_VERIFYEXISTS set,
2160      * according to MSDN: https://msdn.microsoft.com/en-us/library/windows/desktop/bb776478(v=vs.85).aspx
2161      */
2162     static PCWSTR Extensions[] = {L".pif", L".com", L".bat", L".cmd", L".lnk", L".exe", NULL};
2163     #define LNK_EXT_INDEX   4   // ".lnk" has index 4 in the array above
2164 
2165     /*
2166      * Start at the beginning of the list if PRF_EXECUTABLE is set, otherwise
2167      * just use the last element 'NULL' (no extension checking).
2168      */
2169     i = ((fFlags & PRF_EXECUTABLE) ? 0 : _countof(Extensions) - 1);
2170     for (; i < _countof(Extensions); ++i)
2171     {
2172         /* Ignore shell links ".lnk" if needed */
2173         if ((fFlags & PRF_DONTFINDLNK) && (i == LNK_EXT_INDEX))
2174             continue;
2175 
2176         Success = (SearchPathW(NULL, pszPath, Extensions[i],
2177                                _countof(szPath), szPath, NULL) != 0);
2178         if (!Success)
2179         {
2180             ERR("SearchPathW(pszPath = '%S') failed\n", pszPath);
2181         }
2182         else
2183         {
2184             ERR("SearchPathW(pszPath = '%S', szPath = '%S') succeeded\n", pszPath, szPath);
2185             break;
2186         }
2187     }
2188 
2189     if (!Success)
2190     {
2191         ERR("SearchPathW(pszPath = '%S') failed\n", pszPath);
2192 
2193         /* We failed, try with PathFindOnPath, as explained by MSDN */
2194         // Success = PathFindOnPathW(pszPath, dirs);
2195         StringCchCopyW(szPath, _countof(szPath), pszPath);
2196         Success = PathFindOnPathW(szPath, dirs);
2197         if (!Success)
2198         {
2199             ERR("PathFindOnPathW(pszPath = '%S') failed\n", pszPath);
2200 
2201             /* We failed again, fall back to building a possible non-existing path */
2202             if (!GetFullPathNameW(pszPath, _countof(szPath), szPath, &fname))
2203             {
2204                 ERR("GetFullPathNameW(pszPath = '%S') failed\n", pszPath);
2205                 return FALSE;
2206             }
2207 
2208             Success = PathFileExistsW(szPath);
2209             if (!Success)
2210                 ERR("PathFileExistsW(szPath = '%S') failed\n", szPath);
2211 
2212             /******************************************************/
2213             /* Question: Why this line is needed only for files?? */
2214             if (fname && (_wcsicmp(pszPath, fname) == 0))
2215                 *szPath = L'\0';
2216             /******************************************************/
2217         }
2218         else
2219         {
2220             ERR("PathFindOnPathW(pszPath = '%S' ==> '%S') succeeded\n", pszPath, szPath);
2221         }
2222     }
2223 
2224     /* Copy back the results to the caller */
2225     StringCchCopyW(pszPath, MAX_PATH, szPath);
2226 
2227     /*
2228      * Since the called functions always checked whether the file path existed,
2229      * we do not need to redo a final check: we can use instead the cached
2230      * result in 'Success'.
2231      */
2232     return ((fFlags & PRF_VERIFYEXISTS) ? Success : TRUE);
2233 #endif
2234 }
2235 
2236 HRESULT CShellLink::SetTargetFromPIDLOrPath(LPCITEMIDLIST pidl, LPCWSTR pszFile)
2237 {
2238     HRESULT hr = S_OK;
2239     LPITEMIDLIST pidlNew = NULL;
2240     WCHAR szPath[MAX_PATH];
2241 
2242     /*
2243      * Not both 'pidl' and 'pszFile' should be set.
2244      * But either one or both can be NULL.
2245      */
2246     if (pidl && pszFile)
2247         return E_FAIL;
2248 
2249     if (pidl)
2250     {
2251         /* Clone the PIDL */
2252         pidlNew = ILClone(pidl);
2253         if (!pidlNew)
2254             return E_FAIL;
2255     }
2256     else if (pszFile)
2257     {
2258         /* Build a PIDL for this path target */
2259         hr = SHILCreateFromPathW(pszFile, &pidlNew, NULL);
2260         if (FAILED(hr))
2261         {
2262             /* This failed, try to resolve the path, then create a simple PIDL */
2263 
2264             StringCchCopyW(szPath, _countof(szPath), pszFile);
2265             // FIXME: Because PathResolve is unimplemented, we use our hackish implementation!
2266             HACKISH_PathResolve(szPath, NULL, PRF_TRYPROGRAMEXTENSIONS);
2267 
2268             pidlNew = SHSimpleIDListFromPathW(szPath);
2269             /******************************************************/
2270             /* Question: Why this line is needed only for files?? */
2271             hr = (*szPath ? S_OK : E_INVALIDARG); // S_FALSE
2272             /******************************************************/
2273         }
2274     }
2275     // else if (!pidl && !pszFile) { pidlNew = NULL; hr = S_OK; }
2276 
2277     ILFree(m_pPidl);
2278     m_pPidl = pidlNew;
2279 
2280     if (!pszFile)
2281     {
2282         if (SHGetPathFromIDListW(pidlNew, szPath))
2283             pszFile = szPath;
2284     }
2285 
2286     // TODO: Fully update link info, tracker, file attribs...
2287 
2288     // if (pszFile)
2289     if (!pszFile)
2290     {
2291         *szPath = L'\0';
2292         pszFile = szPath;
2293     }
2294 
2295     /* Update the cached path (for link info) */
2296     ShellLink_GetVolumeInfo(pszFile, &volume);
2297     m_sPath = (LPWSTR)HeapAlloc(GetProcessHeap(), 0,
2298                               (wcslen(pszFile) + 1) * sizeof(WCHAR));
2299     if (!m_sPath)
2300         return E_OUTOFMEMORY;
2301     wcscpy(m_sPath, pszFile);
2302 
2303     m_bDirty = TRUE;
2304     return hr;
2305 }
2306 
2307 HRESULT STDMETHODCALLTYPE CShellLink::SetPath(LPCWSTR pszFile)
2308 {
2309     LPWSTR unquoted = NULL;
2310     HRESULT hr = S_OK;
2311 
2312     TRACE("(%p)->(path=%s)\n", this, debugstr_w(pszFile));
2313 
2314     if (!pszFile)
2315         return E_INVALIDARG;
2316 
2317     /*
2318      * Allow upgrading Logo3 shortcuts (m_Header.dwFlags & SLDF_HAS_LOGO3ID),
2319      * but forbid upgrading Darwin ones.
2320      */
2321     if (m_Header.dwFlags & SLDF_HAS_DARWINID)
2322         return S_FALSE;
2323 
2324     /* quotes at the ends of the string are stripped */
2325     SIZE_T len = wcslen(pszFile);
2326     if (pszFile[0] == L'"' && pszFile[len-1] == L'"')
2327     {
2328         unquoted = strdupW(pszFile);
2329         PathUnquoteSpacesW(unquoted);
2330         pszFile = unquoted;
2331     }
2332 
2333     /* any other quote marks are invalid */
2334     if (wcschr(pszFile, L'"'))
2335     {
2336         hr = S_FALSE;
2337         goto end;
2338     }
2339 
2340     /* Clear the cached path */
2341     HeapFree(GetProcessHeap(), 0, m_sPath);
2342     m_sPath = NULL;
2343 
2344     /* Check for an advertised target (Logo3 or Darwin) */
2345     if (SetAdvertiseInfo(pszFile) != S_OK)
2346     {
2347         /* This is not an advertised target, but a regular path */
2348         WCHAR szPath[MAX_PATH];
2349 
2350         /*
2351          * Check whether the user-given file path contains unexpanded
2352          * environment variables. If so, create a target environment block.
2353          * Note that in this block we will store the user-given path.
2354          * It will contain the unexpanded environment variables, but
2355          * it can also contain already expanded path that the user does
2356          * not want to see them unexpanded (e.g. so that they always
2357          * refer to the same place even if the would-be corresponding
2358          * environment variable could change).
2359          */
2360         if (*pszFile)
2361             SHExpandEnvironmentStringsW(pszFile, szPath, _countof(szPath));
2362         else
2363             *szPath = L'\0';
2364 
2365         if (*pszFile && (wcscmp(pszFile, szPath) != 0))
2366         {
2367             /*
2368              * The user-given file path contains unexpanded environment
2369              * variables, so we need a target environment block.
2370              */
2371             EXP_SZ_LINK buffer;
2372             LPEXP_SZ_LINK pInfo;
2373 
2374             pInfo = (LPEXP_SZ_LINK)SHFindDataBlock(m_pDBList, EXP_SZ_LINK_SIG);
2375             if (pInfo)
2376             {
2377                 /* Make sure that the size of the structure is valid */
2378                 if (pInfo->cbSize != sizeof(*pInfo))
2379                 {
2380                     ERR("Ooops. This structure is not as expected...\n");
2381 
2382                     /* Invalid structure, remove it altogether */
2383                     m_Header.dwFlags &= ~SLDF_HAS_EXP_SZ;
2384                     RemoveDataBlock(EXP_SZ_LINK_SIG);
2385 
2386                     /* Reset the pointer and go use the static buffer */
2387                     pInfo = NULL;
2388                 }
2389             }
2390             if (!pInfo)
2391             {
2392                 /* Use the static buffer */
2393                 pInfo = &buffer;
2394                 buffer.cbSize = sizeof(buffer);
2395                 buffer.dwSignature = EXP_SZ_LINK_SIG;
2396             }
2397 
2398             lstrcpynW(pInfo->szwTarget, pszFile, _countof(pInfo->szwTarget));
2399             WideCharToMultiByte(CP_ACP, 0, pszFile, -1,
2400                                 pInfo->szTarget, _countof(pInfo->szTarget), NULL, NULL);
2401 
2402             hr = S_OK;
2403             if (pInfo == &buffer)
2404                 hr = AddDataBlock(pInfo);
2405             if (hr == S_OK)
2406                 m_Header.dwFlags |= SLDF_HAS_EXP_SZ;
2407 
2408             /* Now, make pszFile point to the expanded buffer */
2409             pszFile = szPath;
2410         }
2411         else
2412         {
2413             /*
2414              * The user-given file path does not contain unexpanded environment
2415              * variables, so we need to remove any target environment block.
2416              */
2417             m_Header.dwFlags &= ~SLDF_HAS_EXP_SZ;
2418             RemoveDataBlock(EXP_SZ_LINK_SIG);
2419 
2420             /* pszFile points to the user path */
2421         }
2422 
2423         /* Set the target */
2424         hr = SetTargetFromPIDLOrPath(NULL, pszFile);
2425     }
2426 
2427     m_bDirty = TRUE;
2428 
2429 end:
2430     HeapFree(GetProcessHeap(), 0, unquoted);
2431     return hr;
2432 }
2433 
2434 HRESULT STDMETHODCALLTYPE CShellLink::AddDataBlock(void* pDataBlock)
2435 {
2436     if (SHAddDataBlock(&m_pDBList, (DATABLOCK_HEADER*)pDataBlock))
2437     {
2438         m_bDirty = TRUE;
2439         return S_OK;
2440     }
2441     return S_FALSE;
2442 }
2443 
2444 HRESULT STDMETHODCALLTYPE CShellLink::CopyDataBlock(DWORD dwSig, void** ppDataBlock)
2445 {
2446     DATABLOCK_HEADER* pBlock;
2447     PVOID pDataBlock;
2448 
2449     TRACE("%p %08x %p\n", this, dwSig, ppDataBlock);
2450 
2451     *ppDataBlock = NULL;
2452 
2453     pBlock = SHFindDataBlock(m_pDBList, dwSig);
2454     if (!pBlock)
2455     {
2456         ERR("unknown datablock %08x (not found)\n", dwSig);
2457         return E_FAIL;
2458     }
2459 
2460     pDataBlock = LocalAlloc(LMEM_ZEROINIT, pBlock->cbSize);
2461     if (!pDataBlock)
2462         return E_OUTOFMEMORY;
2463 
2464     CopyMemory(pDataBlock, pBlock, pBlock->cbSize);
2465 
2466     *ppDataBlock = pDataBlock;
2467     return S_OK;
2468 }
2469 
2470 HRESULT STDMETHODCALLTYPE CShellLink::RemoveDataBlock(DWORD dwSig)
2471 {
2472     if (SHRemoveDataBlock(&m_pDBList, dwSig))
2473     {
2474         m_bDirty = TRUE;
2475         return S_OK;
2476     }
2477     return S_FALSE;
2478 }
2479 
2480 HRESULT STDMETHODCALLTYPE CShellLink::GetFlags(DWORD *pdwFlags)
2481 {
2482     TRACE("%p %p\n", this, pdwFlags);
2483     *pdwFlags = m_Header.dwFlags;
2484     return S_OK;
2485 }
2486 
2487 HRESULT STDMETHODCALLTYPE CShellLink::SetFlags(DWORD dwFlags)
2488 {
2489 #if 0 // FIXME!
2490     m_Header.dwFlags = dwFlags;
2491     m_bDirty = TRUE;
2492     return S_OK;
2493 #else
2494     FIXME("\n");
2495     return E_NOTIMPL;
2496 #endif
2497 }
2498 
2499 /**************************************************************************
2500  * CShellLink implementation of IShellExtInit::Initialize()
2501  *
2502  * Loads the shelllink from the dataobject the shell is pointing to.
2503  */
2504 HRESULT STDMETHODCALLTYPE CShellLink::Initialize(LPCITEMIDLIST pidlFolder, IDataObject *pdtobj, HKEY hkeyProgID)
2505 {
2506     TRACE("%p %p %p %p\n", this, pidlFolder, pdtobj, hkeyProgID);
2507 
2508     if (!pdtobj)
2509         return E_FAIL;
2510 
2511     FORMATETC format;
2512     format.cfFormat = CF_HDROP;
2513     format.ptd = NULL;
2514     format.dwAspect = DVASPECT_CONTENT;
2515     format.lindex = -1;
2516     format.tymed = TYMED_HGLOBAL;
2517 
2518     STGMEDIUM stgm;
2519     HRESULT hr = pdtobj->GetData(&format, &stgm);
2520     if (FAILED(hr))
2521         return hr;
2522 
2523     UINT count = DragQueryFileW((HDROP)stgm.hGlobal, -1, NULL, 0);
2524     if (count == 1)
2525     {
2526         count = DragQueryFileW((HDROP)stgm.hGlobal, 0, NULL, 0);
2527         count++;
2528         LPWSTR path = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, count * sizeof(WCHAR));
2529         if (path)
2530         {
2531             count = DragQueryFileW((HDROP)stgm.hGlobal, 0, path, count);
2532             hr = Load(path, 0);
2533             HeapFree(GetProcessHeap(), 0, path);
2534         }
2535     }
2536     ReleaseStgMedium(&stgm);
2537 
2538     return S_OK;
2539 }
2540 
2541 HRESULT STDMETHODCALLTYPE CShellLink::QueryContextMenu(HMENU hMenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags)
2542 {
2543     int id = 1;
2544 
2545     TRACE("%p %p %u %u %u %u\n", this,
2546           hMenu, indexMenu, idCmdFirst, idCmdLast, uFlags);
2547 
2548     if (!hMenu)
2549         return E_INVALIDARG;
2550 
2551     WCHAR wszOpen[20];
2552     if (!LoadStringW(shell32_hInstance, IDS_OPEN_VERB, wszOpen, _countof(wszOpen)))
2553         *wszOpen = L'\0';
2554 
2555     MENUITEMINFOW mii;
2556     ZeroMemory(&mii, sizeof(mii));
2557     mii.cbSize = sizeof(mii);
2558     mii.fMask = MIIM_TYPE | MIIM_ID | MIIM_STATE;
2559     mii.dwTypeData = wszOpen;
2560     mii.cch = wcslen(mii.dwTypeData);
2561     mii.wID = idCmdFirst + id++;
2562     mii.fState = MFS_DEFAULT | MFS_ENABLED;
2563     mii.fType = MFT_STRING;
2564     if (!InsertMenuItemW(hMenu, indexMenu, TRUE, &mii))
2565         return E_FAIL;
2566     m_iIdOpen = 1;
2567 
2568     return MAKE_HRESULT(SEVERITY_SUCCESS, 0, id);
2569 }
2570 
2571 HRESULT STDMETHODCALLTYPE CShellLink::InvokeCommand(LPCMINVOKECOMMANDINFO lpici)
2572 {
2573     LPWSTR args = NULL;
2574     LPWSTR path = NULL;
2575 
2576     TRACE("%p %p\n", this, lpici);
2577 
2578     if (lpici->cbSize < sizeof(CMINVOKECOMMANDINFO))
2579         return E_INVALIDARG;
2580 
2581     // NOTE: We could use lpici->hwnd (certainly in case lpici->fMask doesn't contain CMIC_MASK_FLAG_NO_UI)
2582     // as the parent window handle... ?
2583     /* FIXME: get using interface set from IObjectWithSite?? */
2584     // NOTE: We might need an extended version of Resolve that provides us with paths...
2585     HRESULT hr = Resolve(lpici->hwnd, 0);
2586     if (FAILED(hr))
2587     {
2588         TRACE("failed to resolve component with error 0x%08x", hr);
2589         return hr;
2590     }
2591 
2592     path = strdupW(m_sPath);
2593 
2594     if ( lpici->cbSize == sizeof(CMINVOKECOMMANDINFOEX) &&
2595         (lpici->fMask & CMIC_MASK_UNICODE) )
2596     {
2597         LPCMINVOKECOMMANDINFOEX iciex = (LPCMINVOKECOMMANDINFOEX)lpici;
2598         SIZE_T len = 2;
2599 
2600         if (m_sArgs)
2601             len += wcslen(m_sArgs);
2602         if (iciex->lpParametersW)
2603             len += wcslen(iciex->lpParametersW);
2604 
2605         args = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
2606         *args = 0;
2607         if (m_sArgs)
2608             wcscat(args, m_sArgs);
2609         if (iciex->lpParametersW)
2610         {
2611             wcscat(args, L" ");
2612             wcscat(args, iciex->lpParametersW);
2613         }
2614     }
2615     else if (m_sArgs != NULL)
2616     {
2617         args = strdupW(m_sArgs);
2618     }
2619 
2620     SHELLEXECUTEINFOW sei;
2621     ZeroMemory(&sei, sizeof(sei));
2622     sei.cbSize = sizeof(sei);
2623     sei.fMask = SEE_MASK_HASLINKNAME | SEE_MASK_UNICODE |
2624                (lpici->fMask & (SEE_MASK_NOASYNC | SEE_MASK_ASYNCOK | SEE_MASK_FLAG_NO_UI));
2625     sei.lpFile = path;
2626     sei.lpClass = m_sLinkPath;
2627     sei.nShow = m_Header.nShowCommand;
2628     sei.lpDirectory = m_sWorkDir;
2629     sei.lpParameters = args;
2630     sei.lpVerb = L"open";
2631 
2632     // HACK for ShellExecuteExW
2633     if (m_sPath && wcsstr(m_sPath, L".cpl"))
2634         sei.lpVerb = L"cplopen";
2635 
2636     if (ShellExecuteExW(&sei))
2637         hr = S_OK;
2638     else
2639         hr = E_FAIL;
2640 
2641     HeapFree(GetProcessHeap(), 0, args);
2642     HeapFree(GetProcessHeap(), 0, path);
2643 
2644     return hr;
2645 }
2646 
2647 HRESULT STDMETHODCALLTYPE CShellLink::GetCommandString(UINT_PTR idCmd, UINT uType, UINT* pwReserved, LPSTR pszName, UINT cchMax)
2648 {
2649     FIXME("%p %lu %u %p %p %u\n", this, idCmd, uType, pwReserved, pszName, cchMax);
2650     return E_NOTIMPL;
2651 }
2652 
2653 INT_PTR CALLBACK ExtendedShortcutProc(HWND hwndDlg, UINT uMsg,
2654                                       WPARAM wParam, LPARAM lParam)
2655 {
2656     switch(uMsg)
2657     {
2658         case WM_INITDIALOG:
2659             if (lParam)
2660             {
2661                 HWND hDlgCtrl = GetDlgItem(hwndDlg, 14000);
2662                 SendMessage(hDlgCtrl, BM_SETCHECK, BST_CHECKED, 0);
2663             }
2664             return TRUE;
2665         case WM_COMMAND:
2666         {
2667             HWND hDlgCtrl = GetDlgItem(hwndDlg, 14000);
2668             if (LOWORD(wParam) == IDOK)
2669             {
2670                 if (SendMessage(hDlgCtrl, BM_GETCHECK, 0, 0) == BST_CHECKED)
2671                     EndDialog(hwndDlg, 1);
2672                 else
2673                     EndDialog(hwndDlg, 0);
2674             }
2675             else if (LOWORD(wParam) == IDCANCEL)
2676             {
2677                 EndDialog(hwndDlg, -1);
2678             }
2679             else if (LOWORD(wParam) == 14000)
2680             {
2681                 if (SendMessage(hDlgCtrl, BM_GETCHECK, 0, 0) == BST_CHECKED)
2682                     SendMessage(hDlgCtrl, BM_SETCHECK, BST_UNCHECKED, 0);
2683                 else
2684                     SendMessage(hDlgCtrl, BM_SETCHECK, BST_CHECKED, 0);
2685             }
2686         }
2687     }
2688     return FALSE;
2689 }
2690 
2691 EXTERN_C HRESULT
2692 WINAPI
2693 SHOpenFolderAndSelectItems(LPITEMIDLIST pidlFolder,
2694                            UINT cidl,
2695                            PCUITEMID_CHILD_ARRAY apidl,
2696                            DWORD dwFlags);
2697 
2698 /**************************************************************************
2699 * SH_GetTargetTypeByPath
2700 *
2701 * Function to get target type by passing full path to it
2702 */
2703 LPWSTR SH_GetTargetTypeByPath(LPCWSTR lpcwFullPath)
2704 {
2705     LPCWSTR pwszExt;
2706     static WCHAR wszBuf[MAX_PATH];
2707 
2708     /* Get file information */
2709     SHFILEINFOW fi;
2710     if (!SHGetFileInfoW(lpcwFullPath, 0, &fi, sizeof(fi), SHGFI_TYPENAME | SHGFI_USEFILEATTRIBUTES))
2711     {
2712         ERR("SHGetFileInfoW failed for %ls (%lu)\n", lpcwFullPath, GetLastError());
2713         fi.szTypeName[0] = L'\0';
2714         fi.hIcon = NULL;
2715     }
2716 
2717     pwszExt = PathFindExtensionW(lpcwFullPath);
2718     if (pwszExt[0])
2719     {
2720         if (!fi.szTypeName[0])
2721         {
2722             /* The file type is unknown, so default to string "FileExtension File" */
2723             size_t cchRemaining = 0;
2724             LPWSTR pwszEnd = NULL;
2725 
2726             StringCchPrintfExW(wszBuf, _countof(wszBuf), &pwszEnd, &cchRemaining, 0, L"%s ", pwszExt + 1);
2727         }
2728         else
2729         {
2730             /* Update file type */
2731             StringCbPrintfW(wszBuf, sizeof(wszBuf), L"%s (%s)", fi.szTypeName, pwszExt);
2732         }
2733     }
2734 
2735     return wszBuf;
2736 }
2737 
2738 /**************************************************************************
2739  * SH_ShellLinkDlgProc
2740  *
2741  * dialog proc of the shortcut property dialog
2742  */
2743 
2744 INT_PTR CALLBACK CShellLink::SH_ShellLinkDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
2745 {
2746     CShellLink *pThis = reinterpret_cast<CShellLink *>(GetWindowLongPtr(hwndDlg, DWLP_USER));
2747 
2748     switch(uMsg)
2749     {
2750         case WM_INITDIALOG:
2751         {
2752             LPPROPSHEETPAGEW ppsp = (LPPROPSHEETPAGEW)lParam;
2753             if (ppsp == NULL)
2754                 break;
2755 
2756             TRACE("ShellLink_DlgProc (WM_INITDIALOG hwnd %p lParam %p ppsplParam %x)\n", hwndDlg, lParam, ppsp->lParam);
2757 
2758             pThis = reinterpret_cast<CShellLink *>(ppsp->lParam);
2759             SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pThis);
2760 
2761             TRACE("m_sArgs: %S sComponent: %S m_sDescription: %S m_sIcoPath: %S m_sPath: %S m_sPathRel: %S sProduct: %S m_sWorkDir: %S\n", pThis->m_sArgs, pThis->sComponent, pThis->m_sDescription,
2762                   pThis->m_sIcoPath, pThis->m_sPath, pThis->m_sPathRel, pThis->sProduct, pThis->m_sWorkDir);
2763 
2764             /* Get file information */
2765             // FIXME! FIXME! Shouldn't we use pThis->m_sIcoPath, pThis->m_Header.nIconIndex instead???
2766             SHFILEINFOW fi;
2767             if (!SHGetFileInfoW(pThis->m_sLinkPath, 0, &fi, sizeof(fi), SHGFI_TYPENAME | SHGFI_ICON))
2768             {
2769                 ERR("SHGetFileInfoW failed for %ls (%lu)\n", pThis->m_sLinkPath, GetLastError());
2770                 fi.szTypeName[0] = L'\0';
2771                 fi.hIcon = NULL;
2772             }
2773 
2774             if (fi.hIcon) // TODO: destroy icon
2775                 SendDlgItemMessageW(hwndDlg, 14000, STM_SETICON, (WPARAM)fi.hIcon, 0);
2776             else
2777                 ERR("ExtractIconW failed %ls %u\n", pThis->m_sIcoPath, pThis->m_Header.nIconIndex);
2778 
2779             /* Target type */
2780             if (pThis->m_sPath)
2781                 SetDlgItemTextW(hwndDlg, 14005, SH_GetTargetTypeByPath(pThis->m_sPath));
2782 
2783             /* Target location */
2784             if (pThis->m_sPath)
2785             {
2786                 WCHAR target[MAX_PATH];
2787                 StringCchCopyW(target, _countof(target), pThis->m_sPath);
2788                 PathRemoveFileSpecW(target);
2789                 SetDlgItemTextW(hwndDlg, 14007, PathFindFileNameW(target));
2790             }
2791 
2792             /* Target path */
2793             if (pThis->m_sPath)
2794             {
2795                 WCHAR newpath[2*MAX_PATH] = L"\0";
2796                 if (wcschr(pThis->m_sPath, ' '))
2797                     StringCchPrintfExW(newpath, _countof(newpath), NULL, NULL, 0, L"\"%ls\"", pThis->m_sPath);
2798                 else
2799                     StringCchCopyExW(newpath, _countof(newpath), pThis->m_sPath, NULL, NULL, 0);
2800 
2801                 if (pThis->m_sArgs && pThis->m_sArgs[0])
2802                 {
2803                     StringCchCatW(newpath, _countof(newpath), L" ");
2804                     StringCchCatW(newpath, _countof(newpath), pThis->m_sArgs);
2805                 }
2806                 SetDlgItemTextW(hwndDlg, 14009, newpath);
2807             }
2808 
2809             /* Working dir */
2810             if (pThis->m_sWorkDir)
2811                 SetDlgItemTextW(hwndDlg, 14011, pThis->m_sWorkDir);
2812 
2813             /* Description */
2814             if (pThis->m_sDescription)
2815                 SetDlgItemTextW(hwndDlg, 14019, pThis->m_sDescription);
2816 
2817             return TRUE;
2818         }
2819 
2820         case WM_NOTIFY:
2821         {
2822             LPPSHNOTIFY lppsn = (LPPSHNOTIFY)lParam;
2823             if (lppsn->hdr.code == PSN_APPLY)
2824             {
2825                 WCHAR wszBuf[MAX_PATH];
2826                 /* set working directory */
2827                 GetDlgItemTextW(hwndDlg, 14011, wszBuf, _countof(wszBuf));
2828                 pThis->SetWorkingDirectory(wszBuf);
2829                 /* set link destination */
2830                 GetDlgItemTextW(hwndDlg, 14009, wszBuf, _countof(wszBuf));
2831                 LPWSTR lpszArgs = NULL;
2832                 LPWSTR unquoted = strdupW(wszBuf);
2833                 StrTrimW(unquoted, L" ");
2834                 if (!PathFileExistsW(unquoted))
2835                 {
2836                     lpszArgs = PathGetArgsW(unquoted);
2837                     PathRemoveArgsW(unquoted);
2838                     StrTrimW(lpszArgs, L" ");
2839                 }
2840                 if (unquoted[0] == '"' && unquoted[wcslen(unquoted)-1] == '"')
2841                     PathUnquoteSpacesW(unquoted);
2842 
2843 
2844                 WCHAR *pwszExt = PathFindExtensionW(unquoted);
2845                 if (!wcsicmp(pwszExt, L".lnk"))
2846                 {
2847                     // FIXME load localized error msg
2848                     MessageBoxW(hwndDlg, L"You cannot create a link to a shortcut", L"Error", MB_ICONERROR);
2849                     SetWindowLongPtr(hwndDlg, DWL_MSGRESULT, PSNRET_INVALID_NOCHANGEPAGE);
2850                     return TRUE;
2851                 }
2852 
2853                 if (!PathFileExistsW(unquoted))
2854                 {
2855                     // FIXME load localized error msg
2856                     MessageBoxW(hwndDlg, L"The specified file name in the target box is invalid", L"Error", MB_ICONERROR);
2857                     SetWindowLongPtr(hwndDlg, DWL_MSGRESULT, PSNRET_INVALID_NOCHANGEPAGE);
2858                     return TRUE;
2859                 }
2860 
2861                 pThis->SetPath(unquoted);
2862                 if (lpszArgs)
2863                     pThis->SetArguments(lpszArgs);
2864                 else
2865                     pThis->SetArguments(L"\0");
2866 
2867                 HeapFree(GetProcessHeap(), 0, unquoted);
2868 
2869                 TRACE("This %p m_sLinkPath %S\n", pThis, pThis->m_sLinkPath);
2870                 pThis->Save(pThis->m_sLinkPath, TRUE);
2871                 SetWindowLongPtr(hwndDlg, DWL_MSGRESULT, PSNRET_NOERROR);
2872                 return TRUE;
2873             }
2874             break;
2875         }
2876 
2877         case WM_COMMAND:
2878             switch(LOWORD(wParam))
2879             {
2880                 case 14020:
2881                     SHOpenFolderAndSelectItems(pThis->m_pPidl, 0, NULL, 0);
2882                     ///
2883                     /// FIXME
2884                     /// open target directory
2885                     ///
2886                     return TRUE;
2887 
2888                 case 14021:
2889                 {
2890                     WCHAR wszPath[MAX_PATH] = L"";
2891 
2892                     if (pThis->m_sIcoPath)
2893                         wcscpy(wszPath, pThis->m_sIcoPath);
2894                     INT IconIndex = pThis->m_Header.nIconIndex;
2895                     if (PickIconDlg(hwndDlg, wszPath, _countof(wszPath), &IconIndex))
2896                     {
2897                         pThis->SetIconLocation(wszPath, IconIndex);
2898                         ///
2899                         /// FIXME redraw icon
2900                         ///
2901                     }
2902                     return TRUE;
2903                 }
2904 
2905                 case 14022:
2906                 {
2907                     INT_PTR result = DialogBoxParamW(shell32_hInstance, MAKEINTRESOURCEW(IDD_SHORTCUT_EXTENDED_PROPERTIES), hwndDlg, ExtendedShortcutProc, (LPARAM)pThis->m_bRunAs);
2908                     if (result == 1 || result == 0)
2909                     {
2910                         if (pThis->m_bRunAs != result)
2911                         {
2912                             PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
2913                         }
2914 
2915                         pThis->m_bRunAs = result;
2916                     }
2917                     return TRUE;
2918                 }
2919             }
2920             if (HIWORD(wParam) == EN_CHANGE)
2921                 PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
2922             break;
2923 
2924         default:
2925             break;
2926     }
2927     return FALSE;
2928 }
2929 
2930 /**************************************************************************
2931  * ShellLink_IShellPropSheetExt interface
2932  */
2933 
2934 HRESULT STDMETHODCALLTYPE CShellLink::AddPages(LPFNADDPROPSHEETPAGE pfnAddPage, LPARAM lParam)
2935 {
2936     HPROPSHEETPAGE hPage = SH_CreatePropertySheetPage(IDD_SHORTCUT_PROPERTIES, SH_ShellLinkDlgProc, (LPARAM)this, NULL);
2937     if (hPage == NULL)
2938     {
2939         ERR("failed to create property sheet page\n");
2940         return E_FAIL;
2941     }
2942 
2943     if (!pfnAddPage(hPage, lParam))
2944         return E_FAIL;
2945 
2946     return S_OK;
2947 }
2948 
2949 HRESULT STDMETHODCALLTYPE CShellLink::ReplacePage(UINT uPageID, LPFNADDPROPSHEETPAGE pfnReplacePage, LPARAM lParam)
2950 {
2951     TRACE("(%p) (uPageID %u, pfnReplacePage %p lParam %p\n", this, uPageID, pfnReplacePage, lParam);
2952     return E_NOTIMPL;
2953 }
2954 
2955 HRESULT STDMETHODCALLTYPE CShellLink::SetSite(IUnknown *punk)
2956 {
2957     TRACE("%p %p\n", this, punk);
2958 
2959     m_site = punk;
2960 
2961     return S_OK;
2962 }
2963 
2964 HRESULT STDMETHODCALLTYPE CShellLink::GetSite(REFIID iid, void ** ppvSite)
2965 {
2966     TRACE("%p %s %p\n", this, debugstr_guid(&iid), ppvSite);
2967 
2968     if (m_site == NULL)
2969         return E_FAIL;
2970 
2971     return m_site->QueryInterface(iid, ppvSite);
2972 }
2973 
2974 HRESULT STDMETHODCALLTYPE CShellLink::DragEnter(IDataObject *pDataObject,
2975     DWORD dwKeyState, POINTL pt, DWORD *pdwEffect)
2976 {
2977     TRACE("(%p)->(DataObject=%p)\n", this, pDataObject);
2978     LPCITEMIDLIST pidlLast;
2979     CComPtr<IShellFolder> psf;
2980 
2981     HRESULT hr = SHBindToParent(m_pPidl, IID_PPV_ARG(IShellFolder, &psf), &pidlLast);
2982 
2983     if (SUCCEEDED(hr))
2984     {
2985         hr = psf->GetUIObjectOf(0, 1, &pidlLast, IID_NULL_PPV_ARG(IDropTarget, &m_DropTarget));
2986 
2987         if (SUCCEEDED(hr))
2988             hr = m_DropTarget->DragEnter(pDataObject, dwKeyState, pt, pdwEffect);
2989         else
2990             *pdwEffect = DROPEFFECT_NONE;
2991     }
2992     else
2993         *pdwEffect = DROPEFFECT_NONE;
2994 
2995     return S_OK;
2996 }
2997 
2998 HRESULT STDMETHODCALLTYPE CShellLink::DragOver(DWORD dwKeyState, POINTL pt,
2999     DWORD *pdwEffect)
3000 {
3001     TRACE("(%p)\n", this);
3002     HRESULT hr = S_OK;
3003     if (m_DropTarget)
3004         hr = m_DropTarget->DragOver(dwKeyState, pt, pdwEffect);
3005     return hr;
3006 }
3007 
3008 HRESULT STDMETHODCALLTYPE CShellLink::DragLeave()
3009 {
3010     TRACE("(%p)\n", this);
3011     HRESULT hr = S_OK;
3012     if (m_DropTarget)
3013     {
3014         hr = m_DropTarget->DragLeave();
3015         m_DropTarget.Release();
3016     }
3017 
3018     return hr;
3019 }
3020 
3021 HRESULT STDMETHODCALLTYPE CShellLink::Drop(IDataObject *pDataObject,
3022     DWORD dwKeyState, POINTL pt, DWORD *pdwEffect)
3023 {
3024     TRACE("(%p)\n", this);
3025     HRESULT hr = S_OK;
3026     if (m_DropTarget)
3027         hr = m_DropTarget->Drop(pDataObject, dwKeyState, pt, pdwEffect);
3028 
3029     return hr;
3030 }
3031 
3032 /**************************************************************************
3033  *      IShellLink_ConstructFromFile
3034  */
3035 HRESULT WINAPI IShellLink_ConstructFromPath(WCHAR *path, REFIID riid, LPVOID *ppv)
3036 {
3037     CComPtr<IPersistFile> ppf;
3038     HRESULT hr = CShellLink::_CreatorClass::CreateInstance(NULL, IID_PPV_ARG(IPersistFile, &ppf));
3039     if (FAILED(hr))
3040         return hr;
3041 
3042     hr = ppf->Load(path, 0);
3043     if (FAILED(hr))
3044         return hr;
3045 
3046     return ppf->QueryInterface(riid, ppv);
3047 }
3048 
3049 HRESULT WINAPI IShellLink_ConstructFromFile(IShellFolder * psf, LPCITEMIDLIST pidl, REFIID riid, LPVOID *ppv)
3050 {
3051     WCHAR path[MAX_PATH];
3052     if (!ILGetDisplayNameExW(psf, pidl, path, 0))
3053         return E_FAIL;
3054 
3055     return IShellLink_ConstructFromPath(path, riid, ppv);
3056 }
3057