Skip to content

Commit 61f8ff3

Browse files
mattnalexbrainman
authored andcommitted
windows: change Readlink to handle junction
Related issue golang/go#10150 Change-Id: I38e3e13238624655bb828d501bc865ff8a4312e9 Reviewed-on: https://go-review.googlesource.com/7493 Reviewed-by: Alex Brainman <[email protected]>
1 parent d008cdc commit 61f8ff3

File tree

2 files changed

+33
-11
lines changed

2 files changed

+33
-11
lines changed

windows/syscall_windows.go

+14-5
Original file line numberDiff line numberDiff line change
@@ -962,13 +962,22 @@ func Readlink(path string, buf []byte) (n int, err error) {
962962
}
963963

964964
rdb := (*reparseDataBuffer)(unsafe.Pointer(&rdbbuf[0]))
965-
if uintptr(bytesReturned) < unsafe.Sizeof(*rdb) ||
966-
rdb.ReparseTag != IO_REPARSE_TAG_SYMLINK {
967-
// the path is not a symlink but another type of reparse point
965+
var s string
966+
switch rdb.ReparseTag {
967+
case IO_REPARSE_TAG_SYMLINK:
968+
data := (*symbolicLinkReparseBuffer)(unsafe.Pointer(&rdb.reparseBuffer))
969+
p := (*[0xffff]uint16)(unsafe.Pointer(&data.PathBuffer[0]))
970+
s = UTF16ToString(p[data.PrintNameOffset/2 : (data.PrintNameLength-data.PrintNameOffset)/2])
971+
case IO_REPARSE_TAG_MOUNT_POINT:
972+
data := (*mountPointReparseBuffer)(unsafe.Pointer(&rdb.reparseBuffer))
973+
p := (*[0xffff]uint16)(unsafe.Pointer(&data.PathBuffer[0]))
974+
s = UTF16ToString(p[data.PrintNameOffset/2 : (data.PrintNameLength-data.PrintNameOffset)/2])
975+
default:
976+
// the path is not a symlink or junction but another type of reparse
977+
// point
968978
return -1, syscall.ENOENT
969979
}
970-
971-
s := UTF16ToString((*[0xffff]uint16)(unsafe.Pointer(&rdb.PathBuffer[0]))[:rdb.PrintNameLength/2])
972980
n = copy(buf, []byte(s))
981+
973982
return n, nil
974983
}

windows/ztypes_windows.go

+19-6
Original file line numberDiff line numberDiff line change
@@ -1089,12 +1089,7 @@ type TCPKeepalive struct {
10891089
Interval uint32
10901090
}
10911091

1092-
type reparseDataBuffer struct {
1093-
ReparseTag uint32
1094-
ReparseDataLength uint16
1095-
Reserved uint16
1096-
1097-
// SymbolicLinkReparseBuffer
1092+
type symbolicLinkReparseBuffer struct {
10981093
SubstituteNameOffset uint16
10991094
SubstituteNameLength uint16
11001095
PrintNameOffset uint16
@@ -1103,9 +1098,27 @@ type reparseDataBuffer struct {
11031098
PathBuffer [1]uint16
11041099
}
11051100

1101+
type mountPointReparseBuffer struct {
1102+
SubstituteNameOffset uint16
1103+
SubstituteNameLength uint16
1104+
PrintNameOffset uint16
1105+
PrintNameLength uint16
1106+
PathBuffer [1]uint16
1107+
}
1108+
1109+
type reparseDataBuffer struct {
1110+
ReparseTag uint32
1111+
ReparseDataLength uint16
1112+
Reserved uint16
1113+
1114+
// GenericReparseBuffer
1115+
reparseBuffer byte
1116+
}
1117+
11061118
const (
11071119
FSCTL_GET_REPARSE_POINT = 0x900A8
11081120
MAXIMUM_REPARSE_DATA_BUFFER_SIZE = 16 * 1024
1121+
IO_REPARSE_TAG_MOUNT_POINT = 0xA0000003
11091122
IO_REPARSE_TAG_SYMLINK = 0xA000000C
11101123
SYMBOLIC_LINK_FLAG_DIRECTORY = 0x1
11111124
)

0 commit comments

Comments
 (0)