<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1, minimum-scale=1" />
<meta name="generator" content="pdoc 0.10.0" />
<title>utils.worker_thread API documentation</title>
<meta name="description" content="" />
<link rel="preload stylesheet" as="style" href="https://cdnjs.cloudflare.com/ajax/libs/10up-sanitize.css/11.0.1/sanitize.min.css" integrity="sha256-PK9q560IAAa6WVRRh76LtCaI8pjTJ2z11v0miyNNjrs=" crossorigin>
<link rel="preload stylesheet" as="style" href="https://cdnjs.cloudflare.com/ajax/libs/10up-sanitize.css/11.0.1/typography.min.css" integrity="sha256-7l/o7C8jubJiy74VsKTidCy1yBkRtiUGbVkYBylBqUg=" crossorigin>
<link rel="stylesheet preload" as="style" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/10.1.1/styles/github.min.css" crossorigin>
<style>:root{--highlight-color:#fe9}.flex{display:flex !important}body{line-height:1.5em}#content{padding:20px}#sidebar{padding:30px;overflow:hidden}#sidebar > *:last-child{margin-bottom:2cm}.http-server-breadcrumbs{font-size:130%;margin:0 0 15px 0}#footer{font-size:.75em;padding:5px 30px;border-top:1px solid #ddd;text-align:right}#footer p{margin:0 0 0 1em;display:inline-block}#footer p:last-child{margin-right:30px}h1,h2,h3,h4,h5{font-weight:300}h1{font-size:2.5em;line-height:1.1em}h2{font-size:1.75em;margin:1em 0 .50em 0}h3{font-size:1.4em;margin:25px 0 10px 0}h4{margin:0;font-size:105%}h1:target,h2:target,h3:target,h4:target,h5:target,h6:target{background:var(--highlight-color);padding:.2em 0}a{color:#058;text-decoration:none;transition:color .3s ease-in-out}a:hover{color:#e82}.title code{font-weight:bold}h2[id^="header-"]{margin-top:2em}.ident{color:#900}pre code{background:#f8f8f8;font-size:.8em;line-height:1.4em}code{background:#f2f2f1;padding:1px 4px;overflow-wrap:break-word}h1 code{background:transparent}pre{background:#f8f8f8;border:0;border-top:1px solid #ccc;border-bottom:1px solid #ccc;margin:1em 0;padding:1ex}#http-server-module-list{display:flex;flex-flow:column}#http-server-module-list div{display:flex}#http-server-module-list dt{min-width:10%}#http-server-module-list p{margin-top:0}.toc ul,#index{list-style-type:none;margin:0;padding:0}#index code{background:transparent}#index h3{border-bottom:1px solid #ddd}#index ul{padding:0}#index h4{margin-top:.6em;font-weight:bold}@media (min-width:200ex){#index .two-column{column-count:2}}@media (min-width:300ex){#index .two-column{column-count:3}}dl{margin-bottom:2em}dl dl:last-child{margin-bottom:4em}dd{margin:0 0 1em 3em}#header-classes + dl > dd{margin-bottom:3em}dd dd{margin-left:2em}dd p{margin:10px 0}.name{background:#eee;font-weight:bold;font-size:.85em;padding:5px 10px;display:inline-block;min-width:40%}.name:hover{background:#e0e0e0}dt:target .name{background:var(--highlight-color)}.name > span:first-child{white-space:nowrap}.name.class > span:nth-child(2){margin-left:.4em}.inherited{color:#999;border-left:5px solid #eee;padding-left:1em}.inheritance em{font-style:normal;font-weight:bold}.desc h2{font-weight:400;font-size:1.25em}.desc h3{font-size:1em}.desc dt code{background:inherit}.source summary,.git-link-div{color:#666;text-align:right;font-weight:400;font-size:.8em;text-transform:uppercase}.source summary > *{white-space:nowrap;cursor:pointer}.git-link{color:inherit;margin-left:1em}.source pre{max-height:500px;overflow:auto;margin:0}.source pre code{font-size:12px;overflow:visible}.hlist{list-style:none}.hlist li{display:inline}.hlist li:after{content:',\2002'}.hlist li:last-child:after{content:none}.hlist .hlist{display:inline;padding-left:1em}img{max-width:100%}td{padding:0 .5em}.admonition{padding:.1em .5em;margin-bottom:1em}.admonition-title{font-weight:bold}.admonition.note,.admonition.info,.admonition.important{background:#aef}.admonition.todo,.admonition.versionadded,.admonition.tip,.admonition.hint{background:#dfd}.admonition.warning,.admonition.versionchanged,.admonition.deprecated{background:#fd4}.admonition.error,.admonition.danger,.admonition.caution{background:lightpink}</style>
<style media="screen and (min-width: 700px)">@media screen and (min-width:700px){#sidebar{width:30%;height:100vh;overflow:auto;position:sticky;top:0}#content{width:70%;max-width:100ch;padding:3em 4em;border-left:1px solid #ddd}pre code{font-size:1em}.item .name{font-size:1em}main{display:flex;flex-direction:row-reverse;justify-content:flex-end}.toc ul ul,#index ul{padding-left:1.5em}.toc > ul > li{margin-top:.5em}}</style>
<style media="print">@media print{#sidebar h1{page-break-before:always}.source{display:none}}@media print{*{background:transparent !important;color:#000 !important;box-shadow:none !important;text-shadow:none !important}a[href]:after{content:" (" attr(href) ")";font-size:90%}a[href][title]:after{content:none}abbr[title]:after{content:" (" attr(title) ")"}.ir a:after,a[href^="javascript:"]:after,a[href^="#"]:after{content:""}pre,blockquote{border:1px solid #999;page-break-inside:avoid}thead{display:table-header-group}tr,img{page-break-inside:avoid}img{max-width:100% !important}@page{margin:0.5cm}p,h2,h3{orphans:3;widows:3}h1,h2,h3,h4,h5,h6{page-break-after:avoid}}</style>
<script async src="https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.7/latest.js?config=TeX-AMS_CHTML" integrity="sha256-kZafAc6mZvK3W3v1pHOcUix30OHQN6pU/NO2oFkqZVw=" crossorigin></script>
<script defer src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/10.1.1/highlight.min.js" integrity="sha256-Uv3H6lx7dJmRfRvH8TH6kJD1TSK1aFcwgx+mdg3epi8=" crossorigin></script>
<script>window.addEventListener('DOMContentLoaded', () => hljs.initHighlighting())</script>
</head>
<body>
<main>
<article id="content">
<header>
<h1 class="title">Module <code>utils.worker_thread</code></h1>
</header>
<section id="section-intro">
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">#!/usr/bin/env python3

# Import PyTorch root package import torch
import torch

import threading

from . import buffer


class WorkerThread(threading.Thread):
    &#34;&#34;&#34;Worker thread. It&#39;s goal execute deferred functions.&#34;&#34;&#34;

    def __init__(self):
        threading.Thread.__init__(self)

        self.cmds = buffer.Buffer()

        self.completion_event_lock_event = threading.Lock()  # Be default lock is not acquired
        self.completion_event_lock_event.acquire()           # Acquire lock

    def defered_call(self, f, args):
        &#34;&#34;&#34;
        Deferred execution of function, not blocking.

        Args:
            f (function): The deferred function to execute
            args (tuple): Arguments for function f which are needed to be passed

        Returns:
          None
        &#34;&#34;&#34;
        function_and_args = (f, args)
        self.cmds.pushBack(function_and_args)

    def run(self):
        report_about_completion = False

        while True:
            if report_about_completion:
                if len(self.cmds) == 0:
                    report_about_completion = False
                    self.completion_event_lock_event.release()

            self.cmds.waitForItem()   # Wait for item in a work queue, blocking
            item = self.cmds.front()  # Get item without block 

            if type(item) == str:
                if item == &#34;_STOP_&#34;:
                    return                          # We have obtained command to stop execution
                elif item == &#34;_S_COMPLETE_&#34;:
                    report_about_completion = True  # Somebody waits for reporting once for event
                                                    # for report about finishing execution
            else:
                f, args = item  

                # Execute function possibly within own cuda stream context which will allow to
                # submit work into NVIDIA GPU without waiting previous works for another streams
                # IMPORTANT: right now retValue is ignored
                if hasattr(self, &#34;worker_stream&#34;):
                    with torch.cuda.stream(self.worker_stream):
                        retValue = f(self, *args)
                    # Force waiting for finishing write-back operations in GPU memory for that thread
                    self.worker_stream.synchronize()
                else:
                    retValue = f(self, *args)

            self.cmds.popFront()   # Get rid of from item in the queue

    def synchronize(self):
        &#34;&#34;&#34; Wait until thread process all queued tasks before that moment in CPU.&#34;&#34;&#34;

        if len(self.cmds) == 0:
            return
        self.cmds.pushBack(&#34;_S_COMPLETE_&#34;)
        self.completion_event_lock_event.acquire()

        if hasattr(self, &#34;worker_stream&#34;):
            self.worker_stream.synchronize()

    def stop(self):
        &#34;&#34;&#34; Request thread after processing all queued tasks before that moment complete it&#39;s work.&#34;&#34;&#34;
        self.cmds.pushBack(&#34;_STOP_&#34;)

    def stopAndJoin(self):
        &#34;&#34;&#34; Request thread after processing all queued tasks before that moment complete it&#39;s work.&#34;&#34;&#34;
        self.cmds.pushBack(&#34;_STOP_&#34;)

        self.join()


# Unittests for launch please use: &#34;pytest -v worker_thread.py&#34;
# https://docs.pytest.org/en/stable/getting-started.html
def test_worker_thread():
    th1 = WorkerThread()
    z = 0

    def testf(thread, x, y): 
        nonlocal z
        z = x/y
    th1.start()
    th1.defered_call(testf, (6, 2))
    th1.synchronize()
    assert z == 3

    th1.defered_call(testf, (8, 2))
    th1.stopAndJoin()
    assert z == 4
# ======================================================================================================================</code></pre>
</details>
</section>
<section>
</section>
<section>
</section>
<section>
<h2 class="section-title" id="header-functions">Functions</h2>
<dl>
<dt id="utils.worker_thread.test_worker_thread"><code class="name flex">
<span>def <span class="ident">test_worker_thread</span></span>(<span>)</span>
</code></dt>
<dd>
<div class="desc"></div>
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">def test_worker_thread():
    th1 = WorkerThread()
    z = 0

    def testf(thread, x, y): 
        nonlocal z
        z = x/y
    th1.start()
    th1.defered_call(testf, (6, 2))
    th1.synchronize()
    assert z == 3

    th1.defered_call(testf, (8, 2))
    th1.stopAndJoin()
    assert z == 4</code></pre>
</details>
</dd>
</dl>
</section>
<section>
<h2 class="section-title" id="header-classes">Classes</h2>
<dl>
<dt id="utils.worker_thread.WorkerThread"><code class="flex name class">
<span>class <span class="ident">WorkerThread</span></span>
</code></dt>
<dd>
<div class="desc"><p>Worker thread. It's goal execute deferred functions.</p>
<p>This constructor should always be called with keyword arguments. Arguments are:</p>
<p><em>group</em> should be None; reserved for future extension when a ThreadGroup
class is implemented.</p>
<p><em>target</em> is the callable object to be invoked by the run()
method. Defaults to None, meaning nothing is called.</p>
<p><em>name</em> is the thread name. By default, a unique name is constructed of
the form "Thread-N" where N is a small decimal number.</p>
<p><em>args</em> is the argument tuple for the target invocation. Defaults to ().</p>
<p><em>kwargs</em> is a dictionary of keyword arguments for the target
invocation. Defaults to {}.</p>
<p>If a subclass overrides the constructor, it must make sure to invoke
the base class constructor (Thread.<strong>init</strong>()) before doing anything
else to the thread.</p></div>
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">class WorkerThread(threading.Thread):
    &#34;&#34;&#34;Worker thread. It&#39;s goal execute deferred functions.&#34;&#34;&#34;

    def __init__(self):
        threading.Thread.__init__(self)

        self.cmds = buffer.Buffer()

        self.completion_event_lock_event = threading.Lock()  # Be default lock is not acquired
        self.completion_event_lock_event.acquire()           # Acquire lock

    def defered_call(self, f, args):
        &#34;&#34;&#34;
        Deferred execution of function, not blocking.

        Args:
            f (function): The deferred function to execute
            args (tuple): Arguments for function f which are needed to be passed

        Returns:
          None
        &#34;&#34;&#34;
        function_and_args = (f, args)
        self.cmds.pushBack(function_and_args)

    def run(self):
        report_about_completion = False

        while True:
            if report_about_completion:
                if len(self.cmds) == 0:
                    report_about_completion = False
                    self.completion_event_lock_event.release()

            self.cmds.waitForItem()   # Wait for item in a work queue, blocking
            item = self.cmds.front()  # Get item without block 

            if type(item) == str:
                if item == &#34;_STOP_&#34;:
                    return                          # We have obtained command to stop execution
                elif item == &#34;_S_COMPLETE_&#34;:
                    report_about_completion = True  # Somebody waits for reporting once for event
                                                    # for report about finishing execution
            else:
                f, args = item  

                # Execute function possibly within own cuda stream context which will allow to
                # submit work into NVIDIA GPU without waiting previous works for another streams
                # IMPORTANT: right now retValue is ignored
                if hasattr(self, &#34;worker_stream&#34;):
                    with torch.cuda.stream(self.worker_stream):
                        retValue = f(self, *args)
                    # Force waiting for finishing write-back operations in GPU memory for that thread
                    self.worker_stream.synchronize()
                else:
                    retValue = f(self, *args)

            self.cmds.popFront()   # Get rid of from item in the queue

    def synchronize(self):
        &#34;&#34;&#34; Wait until thread process all queued tasks before that moment in CPU.&#34;&#34;&#34;

        if len(self.cmds) == 0:
            return
        self.cmds.pushBack(&#34;_S_COMPLETE_&#34;)
        self.completion_event_lock_event.acquire()

        if hasattr(self, &#34;worker_stream&#34;):
            self.worker_stream.synchronize()

    def stop(self):
        &#34;&#34;&#34; Request thread after processing all queued tasks before that moment complete it&#39;s work.&#34;&#34;&#34;
        self.cmds.pushBack(&#34;_STOP_&#34;)

    def stopAndJoin(self):
        &#34;&#34;&#34; Request thread after processing all queued tasks before that moment complete it&#39;s work.&#34;&#34;&#34;
        self.cmds.pushBack(&#34;_STOP_&#34;)

        self.join()</code></pre>
</details>
<h3>Ancestors</h3>
<ul class="hlist">
<li>threading.Thread</li>
</ul>
<h3>Methods</h3>
<dl>
<dt id="utils.worker_thread.WorkerThread.defered_call"><code class="name flex">
<span>def <span class="ident">defered_call</span></span>(<span>self, f, args)</span>
</code></dt>
<dd>
<div class="desc"><p>Deferred execution of function, not blocking.</p>
<h2 id="args">Args</h2>
<dl>
<dt><strong><code>f</code></strong> :&ensp;<code>function</code></dt>
<dd>The deferred function to execute</dd>
<dt><strong><code>args</code></strong> :&ensp;<code>tuple</code></dt>
<dd>Arguments for function f which are needed to be passed</dd>
</dl>
<h2 id="returns">Returns</h2>
<p>None</p></div>
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">def defered_call(self, f, args):
    &#34;&#34;&#34;
    Deferred execution of function, not blocking.

    Args:
        f (function): The deferred function to execute
        args (tuple): Arguments for function f which are needed to be passed

    Returns:
      None
    &#34;&#34;&#34;
    function_and_args = (f, args)
    self.cmds.pushBack(function_and_args)</code></pre>
</details>
</dd>
<dt id="utils.worker_thread.WorkerThread.run"><code class="name flex">
<span>def <span class="ident">run</span></span>(<span>self)</span>
</code></dt>
<dd>
<div class="desc"><p>Method representing the thread's activity.</p>
<p>You may override this method in a subclass. The standard run() method
invokes the callable object passed to the object's constructor as the
target argument, if any, with sequential and keyword arguments taken
from the args and kwargs arguments, respectively.</p></div>
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">def run(self):
    report_about_completion = False

    while True:
        if report_about_completion:
            if len(self.cmds) == 0:
                report_about_completion = False
                self.completion_event_lock_event.release()

        self.cmds.waitForItem()   # Wait for item in a work queue, blocking
        item = self.cmds.front()  # Get item without block 

        if type(item) == str:
            if item == &#34;_STOP_&#34;:
                return                          # We have obtained command to stop execution
            elif item == &#34;_S_COMPLETE_&#34;:
                report_about_completion = True  # Somebody waits for reporting once for event
                                                # for report about finishing execution
        else:
            f, args = item  

            # Execute function possibly within own cuda stream context which will allow to
            # submit work into NVIDIA GPU without waiting previous works for another streams
            # IMPORTANT: right now retValue is ignored
            if hasattr(self, &#34;worker_stream&#34;):
                with torch.cuda.stream(self.worker_stream):
                    retValue = f(self, *args)
                # Force waiting for finishing write-back operations in GPU memory for that thread
                self.worker_stream.synchronize()
            else:
                retValue = f(self, *args)

        self.cmds.popFront()   # Get rid of from item in the queue</code></pre>
</details>
</dd>
<dt id="utils.worker_thread.WorkerThread.stop"><code class="name flex">
<span>def <span class="ident">stop</span></span>(<span>self)</span>
</code></dt>
<dd>
<div class="desc"><p>Request thread after processing all queued tasks before that moment complete it's work.</p></div>
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">def stop(self):
    &#34;&#34;&#34; Request thread after processing all queued tasks before that moment complete it&#39;s work.&#34;&#34;&#34;
    self.cmds.pushBack(&#34;_STOP_&#34;)</code></pre>
</details>
</dd>
<dt id="utils.worker_thread.WorkerThread.stopAndJoin"><code class="name flex">
<span>def <span class="ident">stopAndJoin</span></span>(<span>self)</span>
</code></dt>
<dd>
<div class="desc"><p>Request thread after processing all queued tasks before that moment complete it's work.</p></div>
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">def stopAndJoin(self):
    &#34;&#34;&#34; Request thread after processing all queued tasks before that moment complete it&#39;s work.&#34;&#34;&#34;
    self.cmds.pushBack(&#34;_STOP_&#34;)

    self.join()</code></pre>
</details>
</dd>
<dt id="utils.worker_thread.WorkerThread.synchronize"><code class="name flex">
<span>def <span class="ident">synchronize</span></span>(<span>self)</span>
</code></dt>
<dd>
<div class="desc"><p>Wait until thread process all queued tasks before that moment in CPU.</p></div>
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">def synchronize(self):
    &#34;&#34;&#34; Wait until thread process all queued tasks before that moment in CPU.&#34;&#34;&#34;

    if len(self.cmds) == 0:
        return
    self.cmds.pushBack(&#34;_S_COMPLETE_&#34;)
    self.completion_event_lock_event.acquire()

    if hasattr(self, &#34;worker_stream&#34;):
        self.worker_stream.synchronize()</code></pre>
</details>
</dd>
</dl>
</dd>
</dl>
</section>
</article>
<nav id="sidebar">
<h1>Index</h1>
<div class="toc">
<ul></ul>
</div>
<ul id="index">
<li><h3><a href="#header-functions">Functions</a></h3>
<ul class="">
<li><code><a title="utils.worker_thread.test_worker_thread" href="#utils.worker_thread.test_worker_thread">test_worker_thread</a></code></li>
</ul>
</li>
<li><h3><a href="#header-classes">Classes</a></h3>
<ul>
<li>
<h4><code><a title="utils.worker_thread.WorkerThread" href="#utils.worker_thread.WorkerThread">WorkerThread</a></code></h4>
<ul class="">
<li><code><a title="utils.worker_thread.WorkerThread.defered_call" href="#utils.worker_thread.WorkerThread.defered_call">defered_call</a></code></li>
<li><code><a title="utils.worker_thread.WorkerThread.run" href="#utils.worker_thread.WorkerThread.run">run</a></code></li>
<li><code><a title="utils.worker_thread.WorkerThread.stop" href="#utils.worker_thread.WorkerThread.stop">stop</a></code></li>
<li><code><a title="utils.worker_thread.WorkerThread.stopAndJoin" href="#utils.worker_thread.WorkerThread.stopAndJoin">stopAndJoin</a></code></li>
<li><code><a title="utils.worker_thread.WorkerThread.synchronize" href="#utils.worker_thread.WorkerThread.synchronize">synchronize</a></code></li>
</ul>
</li>
</ul>
</li>
</ul>
</nav>
</main>
<footer id="footer">
FL_PyTorch. The document generated 06-March-2023 18:59:01.
<p>Generated by <a href="https://pdoc3.github.io/pdoc" title="pdoc: Python API documentation generator"><cite>pdoc</cite> 0.10.0</a>.</p>
</footer>
</body>
</html>