<!DOCTYPE html>
<html lang="en">
	<head>
		<title>portlock - portlock.go</title>
		<link type="text/css" rel="stylesheet" href="/style/src.css">
	</head>
	<body>
		<h1><a href="/portlock">portlock</a> - portlock.go</h1>
		<pre>
<span class="hidden"><a id="L1" href="#L1">     1</a>  </span><span class="text">// Package portlock is a simple mutex for use between processes to protect a shared resource.</span>
<span class="hidden"><a id="L2" href="#L2">     2</a>  </span><span>package portlock </span><span class="text">// import &#34;vimagination.zapto.org/portlock&#34;</span>
<span class="hidden"><a id="L3" href="#L3">     3</a>  </span>
<span class="hidden"><a id="L4" href="#L4">     4</a>  </span><span>import (</span>
<span class="hidden"><a id="L5" href="#L5">     5</a>  </span><span>	&#34;errors&#34;</span>
<span class="hidden"><a id="L6" href="#L6">     6</a>  </span><span>	&#34;io&#34;</span>
<span class="hidden"><a id="L7" href="#L7">     7</a>  </span><span>	&#34;net&#34;</span>
<span class="hidden"><a id="L8" href="#L8">     8</a>  </span><span>	&#34;sync&#34;</span>
<span class="hidden"><a id="L9" href="#L9">     9</a>  </span><span>)</span>
<span class="hidden"><a id="L10" href="#L10">    10</a>  </span>
<span class="hidden"><a id="L11" href="#L11">    11</a>  </span><span class="text">// Mutex is a mutual exclusion lock that can be used across different processes.</span>
<span class="hidden"><a id="L12" href="#L12">    12</a>  </span><span>type mutex struct {</span>
<span class="hidden"><a id="L13" href="#L13">    13</a>  </span><span>	addr string</span>
<span class="hidden"><a id="L14" href="#L14">    14</a>  </span>
<span class="hidden"><a id="L15" href="#L15">    15</a>  </span><span>	mu sync.Mutex</span>
<span class="hidden"><a id="L16" href="#L16">    16</a>  </span><span>	l  io.Closer</span>
<span class="hidden"><a id="L17" href="#L17">    17</a>  </span><span>}</span>
<span class="hidden"><a id="L18" href="#L18">    18</a>  </span>
<span class="hidden"><a id="L19" href="#L19">    19</a>  </span><span>var readBuf [1]byte</span>
<span class="hidden"><a id="L20" href="#L20">    20</a>  </span>
<span class="hidden"><a id="L21" href="#L21">    21</a>  </span><span class="text">// Type Locker combines the sync.Locker interface with the TryLock method.</span>
<span class="hidden"><a id="L22" href="#L22">    22</a>  </span><span>type Locker interface {</span>
<span class="hidden"><a id="L23" href="#L23">    23</a>  </span><span>	sync.Locker</span>
<span class="hidden"><a id="L24" href="#L24">    24</a>  </span><span>	TryLock() bool</span>
<span class="hidden"><a id="L25" href="#L25">    25</a>  </span><span>}</span>
<span class="hidden"><a id="L26" href="#L26">    26</a>  </span>
<span class="hidden"><a id="L27" href="#L27">    27</a>  </span><span class="text">// New creates a new Mutex which currently uses a TCP connection to determine</span>
<span class="hidden"><a id="L28" href="#L28">    28</a>  </span><span class="text">// the lock status, and as such requires a TCP address to listen on.</span>
<span class="hidden"><a id="L29" href="#L29">    29</a>  </span><span class="text">//</span>
<span class="hidden"><a id="L30" href="#L30">    30</a>  </span><span class="text">// This may change and is not stable.</span>
<span class="hidden"><a id="L31" href="#L31">    31</a>  </span><span>func New(addr string) Locker {</span>
<span class="hidden"><a id="L32" href="#L32">    32</a>  </span><span>	return &amp;mutex{addr: addr}</span>
<span class="hidden"><a id="L33" href="#L33">    33</a>  </span><span>}</span>
<span class="hidden"><a id="L34" href="#L34">    34</a>  </span>
<span class="hidden"><a id="L35" href="#L35">    35</a>  </span><span class="text">// Lock locks the mutex. If it is already locked, by this or another process,</span>
<span class="hidden"><a id="L36" href="#L36">    36</a>  </span><span class="text">// then the call blocks until it is unlocked.</span>
<span class="hidden"><a id="L37" href="#L37">    37</a>  </span><span>func (m *mutex) Lock() {</span>
<span class="hidden"><a id="L38" href="#L38">    38</a>  </span><span>	for !m.TryLock() {</span>
<span class="hidden"><a id="L39" href="#L39">    39</a>  </span><span>		if c, err := net.Dial(&#34;tcp&#34;, m.addr); err == nil {</span>
<span class="hidden"><a id="L40" href="#L40">    40</a>  </span><span>			</span><span class="text">// c.SetDeadline(time.Now().Add(time.Second &gt;&gt; 1))</span>
<span class="hidden"><a id="L41" href="#L41">    41</a>  </span><span>			c.Read(readBuf[:])</span>
<span class="hidden"><a id="L42" href="#L42">    42</a>  </span><span>		}</span>
<span class="hidden"><a id="L43" href="#L43">    43</a>  </span><span>	}</span>
<span class="hidden"><a id="L44" href="#L44">    44</a>  </span><span>}</span>
<span class="hidden"><a id="L45" href="#L45">    45</a>  </span>
<span class="hidden"><a id="L46" href="#L46">    46</a>  </span><span class="text">// TryLock attempts to lock the Mutex, returning true on a success.</span>
<span class="hidden"><a id="L47" href="#L47">    47</a>  </span><span>func (m *mutex) TryLock() bool {</span>
<span class="hidden"><a id="L48" href="#L48">    48</a>  </span><span>	var oe *net.OpError</span>
<span class="hidden"><a id="L49" href="#L49">    49</a>  </span>
<span class="hidden"><a id="L50" href="#L50">    50</a>  </span><span>	if l, err := net.Listen(&#34;tcp&#34;, m.addr); err == nil {</span>
<span class="hidden"><a id="L51" href="#L51">    51</a>  </span><span>		m.mu.Lock()</span>
<span class="hidden"><a id="L52" href="#L52">    52</a>  </span><span>		defer m.mu.Unlock()</span>
<span class="hidden"><a id="L53" href="#L53">    53</a>  </span>
<span class="hidden"><a id="L54" href="#L54">    54</a>  </span><span>		m.l = l</span>
<span class="hidden"><a id="L55" href="#L55">    55</a>  </span>
<span class="hidden"><a id="L56" href="#L56">    56</a>  </span><span>		return true</span>
<span class="hidden"><a id="L57" href="#L57">    57</a>  </span><span>	} else if errors.As(err, &amp;oe) &amp;&amp; isOpen(oe.Err) {</span>
<span class="hidden"><a id="L58" href="#L58">    58</a>  </span><span>		return false</span>
<span class="hidden"><a id="L59" href="#L59">    59</a>  </span><span>	} else {</span>
<span class="hidden"><a id="L60" href="#L60">    60</a>  </span><span>		panic(err)</span>
<span class="hidden"><a id="L61" href="#L61">    61</a>  </span><span>	}</span>
<span class="hidden"><a id="L62" href="#L62">    62</a>  </span><span>}</span>
<span class="hidden"><a id="L63" href="#L63">    63</a>  </span>
<span class="hidden"><a id="L64" href="#L64">    64</a>  </span><span class="text">// Unlock removes the lock. Due to the current implementation, exiting the</span>
<span class="hidden"><a id="L65" href="#L65">    65</a>  </span><span class="text">// program will also unlock the mutex.</span>
<span class="hidden"><a id="L66" href="#L66">    66</a>  </span><span class="text">//</span>
<span class="hidden"><a id="L67" href="#L67">    67</a>  </span><span class="text">// It is the intention that this will always be true, but Unlock should be</span>
<span class="hidden"><a id="L68" href="#L68">    68</a>  </span><span class="text">// called before program exit regardless.</span>
<span class="hidden"><a id="L69" href="#L69">    69</a>  </span><span>func (m *mutex) Unlock() {</span>
<span class="hidden"><a id="L70" href="#L70">    70</a>  </span><span>	m.mu.Lock()</span>
<span class="hidden"><a id="L71" href="#L71">    71</a>  </span><span>	defer m.mu.Unlock()</span>
<span class="hidden"><a id="L72" href="#L72">    72</a>  </span>
<span class="hidden"><a id="L73" href="#L73">    73</a>  </span><span>	m.l.Close()</span>
<span class="hidden"><a id="L74" href="#L74">    74</a>  </span>
<span class="hidden"><a id="L75" href="#L75">    75</a>  </span><span>	m.l = nil</span>
<span class="hidden"><a id="L76" href="#L76">    76</a>  </span><span>}</span>
<span class="hidden"><a id="L77" href="#L77">    77</a>  </span></pre>
	</body>
</html>
