<?xml version="1.0" encoding="UTF-8"?>
<?xml-stylesheet type="text/xsl" href="default.xsl"?>
<fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="true"
expanded="true"
root="true"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>365</fr:anchor><fr:addr
type="user">ocs-0001</fr:addr><fr:route>index.xml</fr:route><fr:title
text="Owen">Owen</fr:title><fr:authors><fr:author>owensmith</fr:author></fr:authors><fr:meta
name="author">false</fr:meta></fr:frontmatter><fr:mainmatter><fr:p>I am a <fr:em>Sr. Scientific Software Developer</fr:em> for <fr:link
type="external"
href="https://www.earthdata.nasa.gov/centers/gesdisc-daac">NASA Goddard's Earth Sciences Data Information Services Center</fr:link> and a <fr:em>Ph.D. Candidate</fr:em> in <fr:link
type="external"
href="https://cnr.ncsu.edu/geospatial/">Geospatial Analytics @ NCSU</fr:link>.
You can also find me elsewhere at <fr:link
type="external"
href="https://bsky.app/profile/bean.computer">bsky</fr:link>, <fr:link
type="external"
href="https://git.sr.ht/~obeancomputer/">sourcehut</fr:link>, <fr:link
type="external"
href="https://github.com/ocsmit">github</fr:link>.</fr:p><fr:p>This website is built using <fr:link
type="external"
href="https://www.forester-notes.org">Forester</fr:link> and is used to collect various notes. You can search the site using <fr:code>ctrl+k</fr:code></fr:p><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="false"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>364</fr:anchor><fr:addr
type="user">ocs-0002</fr:addr><fr:route>ocs-0002.xml</fr:route><fr:title
text="Notes">Notes</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>This is a notebook in which I keep notes of a variety of subjects at varying degrees of disjointedness.</fr:p><fr:tree
toc="false"
numbered="false"
show-heading="true"
show-metadata="true"
expanded="false"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>249</fr:anchor><fr:addr
type="user">ocs-0006</fr:addr><fr:route>ocs-0006.xml</fr:route><fr:title
text="Double Logistic Greendown Model">Double Logistic Greendown Model</fr:title><fr:taxon>Note</fr:taxon><fr:date><fr:year>2025</fr:year><fr:month>3</fr:month><fr:day>20</fr:day></fr:date><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Logistic sigmoid function for modeling <fr:link
type="local"
href="ocs-0007.xml"
addr="ocs-0007"
title="Phenology">Phenology</fr:link> detailed in <fr:link
type="local"
href="elmore_ea_2011.xml"
addr="elmore_ea_2011"
title="Landscape controls on the timing of spring, autumn, and growing season length in mid-Atlantic forests">Elmore et. al 2011</fr:link> &amp; <fr:link
type="local"
href="gao_ea_2021.xml"
addr="gao_ea_2021"
title="Long-term, medium spatial resolution annual land surface phenology with a Bayesian hierarchical model">Gao et al. 2011</fr:link>. The function combines spring and autumn seasons into a single equation to account for the <fr:strong>greendown</fr:strong> phenomena by allowing for a gradual reduction of VI values in mid-summer. Typically the model is fit to all VI observations within a single year using maximum likelihood methods.</fr:p><fr:tex
display="block"><![CDATA[ 	v(t, M) = m_1 + (m_2 - m_7t)\left (\frac {1}{1+e^{(m_3-t)/m_4}}-\frac {1}{1 + e^{(m_5-t)/m_6}}\right ) ]]></fr:tex><fr:p>	where
		<fr:ul><fr:li><fr:tex
display="inline"><![CDATA[v(t,m)]]></fr:tex> is the modeled VI value at time <fr:tex
display="inline"><![CDATA[t]]></fr:tex> in terms of day of year (DOY)</fr:li>
			<fr:li><fr:tex
display="inline"><![CDATA[M = [m_1, \dots  , m_7]^T]]></fr:tex> is a parameter vector controlling the shape of the double-logistic function.</fr:li>
			<fr:li><fr:tex
display="inline"><![CDATA[m_1]]></fr:tex> represents the mean VI value in the dormant period;</fr:li>
			<fr:li><fr:tex
display="inline"><![CDATA[(m_2 − m_7t)]]></fr:tex> represents the <fr:strong>difference</fr:strong> between VI value at time <fr:tex
display="inline"><![CDATA[t]]></fr:tex> and the dormant mean;</fr:li>
			<fr:li><fr:tex
display="inline"><![CDATA[m_3]]></fr:tex> and <fr:tex
display="inline"><![CDATA[m_5]]></fr:tex> are the inflection points that are commonly used to represent start-of-season (SOS) and end-of-season (EOS) dates in terms of DOY in the spring and autumn;</fr:li>
			<fr:li><fr:tex
display="inline"><![CDATA[m_4]]></fr:tex> and <fr:tex
display="inline"><![CDATA[m_6]]></fr:tex> are the corresponding slopes of the VI trajectories in spring and autumn;</fr:li>
			<fr:li><fr:tex
display="inline"><![CDATA[m_7]]></fr:tex> is the “greendown” parameter that accounts for the VI greendown phenomenon in the mid-summer time.</fr:li></fr:ul></fr:p><fr:p>Given the set <fr:tex
display="inline"><![CDATA[t \in  [1, \dots , 366]]]></fr:tex> and <fr:tex
display="inline"><![CDATA[M = \{0.05, 1, 120, 8, 290, 6, 0.0011\}]]></fr:tex>, a single years <fr:link
type="local"
href="ocs-0007.xml"
addr="ocs-0007"
title="Phenology">phenology</fr:link> would be estimated by the following curve.</fr:p>
 
  
  <fr:figure><fr:resource
hash="c346ce0392b9a0ec33885eff7e0fa5eb"><fr:resource-content><fr:img
src="data:image/svg+xml;base64,<?xml version='1.0' encoding='UTF-8'?>
<!-- This file was generated by dvisvgm 3.2.2 -->
<svg version='1.1' xmlns='http://www.w3.org/2000/svg' xmlns:xlink='http://www.w3.org/1999/xlink' width='666.57346pt' height='550.378595pt' viewBox='-72 -72 444.382306 366.919064'>
<style type='text/css'>
<![CDATA[@font-face{font-family:cmsy5;src:url(data:application/x-font-woff;base64,d09GRgABAAAAAAL4AAoAAAAAA1gAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABPUy8yAAAA9AAAAEoAAABgql2Vr2NtYXAAAAFAAAAAKgAAADQADCJlZ2x5ZgAAAWwAAAA0AAAANGfP+DpoZWFkAAABoAAAAC0AAAA2K2/zt2hoZWEAAAHQAAAAHgAAACQHgQCRaG10eAAAAfAAAAAIAAAACAgFAKVsb2NhAAAB+AAAAAYAAAAGABoAAG1heHAAAAIAAAAAFwAAACAABAANbmFtZQAAAhgAAADOAAABOBN9Krdwb3N0AAAC6AAAABAAAAAgAAMAAXjaY2BmCWWcwMDKwMA0i+nsPxGGfhDN+IbxDYMxIycDELAxoAAHGCOluCydgU1JSEmIWeuPJQMDsxbzcaAwI0Lp//8MDADpeg5BAAB42mNgYGBiYGBgBmIRIMkIplkYFIA0CxAC+UpC//9DyLvvwfIMAFGoBpAAAAABAKUA4gOwAR4ACwAAJSEGJjQ2FyE2FhQGA4X9Sw0eHg0CtQ0eHuMBCigKAQEKKAoAAAB42mNgZGBgAOLHf36ujue3+crAzMIAAk+YIs4h0yyhjHJAioOBCcQDAElOCcYAAAB42mNgZGBg1vpjycDAEsqwFMjewAAUQQFMAE6hAxgAAARVAAADsAClAAAAAAAaAAB42mNgZGBgYGLgYQDRDFASCQAAAckAEgB42k2MPW/CMBCGn4gEiTCwVB06gAfmCEJZuzAwsTKxADYoQxxkh0iRqo793b24HjjJp+f9OAM5PyQMkzALe5iUTNQ/Z0x5izwWtYic88GSEUk6EWfOV+RUklPkjHdukcfif0fO+eR319SPZ2ucOjTaOHutfb89Guerxqp1sQp6b6xx59ZodemV7irf3Wu1KcqiZEdDzYMnLQaH4iCODmy5Subp2XIMjqeS1EprTcHqJd9LbkPnHH7S0rlIooQ6ufKy79JWbOSyHN4f8HMwSQAAeNpjYGZABowMaAAAAI4ABQ==) format('woff');}
@font-face{font-family:cmmi5;src:url(data:application/x-font-woff;base64,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) format('woff');}
@font-face{font-family:cmr5;src:url(data:application/x-font-woff;base64,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) format('woff');}
text.f0 {font-family:cmsy5;font-size:4.98132px}
text.f1 {font-family:cmmi5;font-size:4.98132px}
text.f2 {font-family:cmr5;font-size:4.98132px}
]]>
</style>
<g id='page1'>
<g stroke-miterlimit='10' transform='translate(-16.549436,251.1749)scale(0.996264,-0.996264)'>
<g fill='#000' stroke='#000'>
<g stroke-width='0.4'>
<g transform='matrix(2.0,0.0,0.0,2.0,0.0,0.0)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(-16.549436,251.1749)scale(-1,-1)'/>
</g>
<g transform='matrix(2.0,0.0,0.0,2.0,-55.65851,-43.9082)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(-16.549436,251.1749)scale(-1,-1)'>
<g stroke='#000' stroke-miterlimit='10' transform='translate(11.175849,229.302819)scale(0.996264,-0.996264)'>
<g fill='#000' stroke='#000'>
<g stroke-width='0.4'>
<g stroke-width='0.2'>
<g fill='#808080' stroke='#808080'>
<path d='M0 0V4.26761M48.74878 0V4.26761M97.49756 0V4.26761M146.24634 0V4.26761' fill='none'/>
</g>
</g>
<g stroke-width='0.2'>
<g fill='#808080' stroke='#808080'>
<path d='M0 0H4.26765M0 35.99854H4.26765M0 71.99707H4.26765M0 107.9956H4.26765M0 143.99414H4.26765' fill='none'/>
</g>
</g>
<path d='M0 0V161.99341H194.99512V0H0Z' fill='none'/>
<g transform='translate(-1.70142,-6.75517)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f2' x='11.175849' y='229.302819'>0</text>
</g>
</g>
</g>
</g>
<g transform='translate(43.64453,-6.75517)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f2' x='11.175849' y='229.302819'>100</text>
</g>
</g>
</g>
</g>
<g transform='translate(92.39331,-6.75517)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f2' x='11.175849' y='229.302819'>200</text>
</g>
</g>
</g>
</g>
<g transform='translate(141.14209,-6.75517)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f2' x='11.175849' y='229.302819'>300</text>
</g>
</g>
</g>
</g>
<g transform='translate(-6.9358,-1.6111)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f2' x='11.175849' y='229.302819'>0</text>
</g>
</g>
</g>
</g>
<g transform='translate(-12.63033,34.38744)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f2' x='11.175849' y='229.302819'>0</text>
<text class='f1' x='14.56597' y='229.302819'>.</text>
<text class='f2' x='16.849097' y='229.302819'>2</text>
</g>
</g>
</g>
</g>
<g transform='translate(-12.63033,70.38597)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f2' x='11.175849' y='229.302819'>0</text>
<text class='f1' x='14.56597' y='229.302819'>.</text>
<text class='f2' x='16.849097' y='229.302819'>4</text>
</g>
</g>
</g>
</g>
<g transform='translate(-12.63033,106.3845)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f2' x='11.175849' y='229.302819'>0</text>
<text class='f1' x='14.56597' y='229.302819'>.</text>
<text class='f2' x='16.849097' y='229.302819'>6</text>
</g>
</g>
</g>
</g>
<g transform='translate(-12.63033,142.38304)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f2' x='11.175849' y='229.302819'>0</text>
<text class='f1' x='14.56597' y='229.302819'>.</text>
<text class='f2' x='16.849097' y='229.302819'>8</text>
</g>
</g>
</g>
</g>
<clipPath id='pgfcp1'>
<path d='M0 0H194.99512V161.99341H0Z'/> </clipPath>
<g clip-path='url(#pgfcp1)'>
<g fill='#000' stroke='#000'>
<path d='M0 8.99968L.53424 8.9997H1.06848L1.60274 8.99971L2.13698 8.99973L2.67123 8.99974L3.20547 8.99976L3.73972 8.99977L4.27397 8.99979L4.80823 8.99982L5.34247 8.99985L5.87671 8.99988L6.41096 8.99991L6.9452 8.99995L7.47946 9L8.0137 9.00005L8.54796 9.0001L9.0822 9.00018L9.61646 9.00026L10.1507 9.00035L10.68494 9.00046L11.2192 9.00058L11.75343 9.00072L12.28769 9.00087L12.82193 9.00105L13.35619 9.00127L13.89043 9.00151L14.42467 9.00177L14.95892 9.00209L15.49318 9.00246L16.02742 9.00285L16.56166 9.00333L17.09592 9.00386L17.63016 9.00449L18.16441 9.00519L18.69865 9.006L19.2329 9.00693L19.76715 9.008L20.3014 9.0092L20.83565 9.01059L21.36989 9.01219L21.90414 9.014L22.43839 9.01611L22.97264 9.01851L23.50688 9.02126L24.04114 9.02438L24.57538 9.02798L25.10962 9.03212L25.64388 9.03683L26.17812 9.04224L26.71237 9.04843L27.24661 9.05553L27.78085 9.06364L28.31511 9.07295L28.84937 9.08359L29.3836 9.0958L29.91785 9.10974L30.4521 9.12575L30.98636 9.14407L31.5206 9.16501L32.05484 9.18906L32.5891 9.21654L33.12334 9.24802L33.6576 9.28404L34.19183 9.32529L34.72607 9.37251L35.26033 9.42656L35.79457 9.48848L36.32883 9.55916L36.86307 9.64018L37.39732 9.73286L37.93156 9.83888L38.4658 9.96004L39.00006 10.09863L39.53432 10.25708L40.06856 10.4383L40.6028 10.64528L41.13705 10.88116L41.6713 11.15114L42.20555 11.45917L42.73979 11.81027L43.27405 12.21028L43.80829 12.666L44.34254 13.18495L44.87679 13.77512L45.41103 14.44545L45.94528 15.20543L46.47952 16.06711L47.01378 17.04211L47.54802 18.14319L48.08228 19.38374L48.61652 20.7802L49.15076 22.34775L49.68501 24.10374L50.21925 26.06047L50.7535 28.2335L51.28775 30.6457L51.82199 33.30556L52.35623 36.2247L52.89049 39.41162L53.42473 42.8746L53.95897 46.61414L54.49323 50.62268L55.02747 54.88963L55.5617 59.40442L56.09596 64.12982L56.6302 69.04767L57.16446 74.11006L57.6987 79.28032L58.23294 84.511L58.7672 89.74254L59.30144 94.9421L59.83568 100.05058L60.36993 105.03256L60.90417 109.83257L61.43842 114.42348L61.97267 118.77353L62.50691 122.8559L63.04115 126.67468L63.57541 130.19707L64.10965 133.43199L64.64389 136.38672L65.17815 139.05319L65.71239 141.47554L66.24663 143.62566L66.78088 145.54752L67.31512 147.24507L67.84938 148.73146L68.38362 150.0365L68.91786 151.17154L69.45212 152.15251L69.98636 152.99518L70.5206 153.71834L71.05486 154.32399L71.5891 154.8337L72.12334 155.25244L72.6576 155.59564L73.19183 155.87263L73.72607 156.0837L74.26033 156.24106L74.79457 156.35548L75.32881 156.43011L75.86307 156.46217L76.39731 156.46518L76.93155 156.43777L77.4658 156.392L78.00005 156.32205L78.53429 156.23093L79.06854 156.12317L79.60278 156.00323L80.13704 155.87276L80.67128 155.72876L81.20552 155.57574L81.73978 155.41228L82.27402 155.24857L82.80826 155.07014L83.34251 154.89305L83.87675 154.7071L84.411 154.51959L84.94525 154.32614L85.47949 154.1312L86.01373 153.93474L86.54799 153.731L87.08223 153.52718L87.61647 153.32622L88.15073 153.11513L88.68497 152.90979L89.21921 152.69865L89.75346 152.48605L90.2877 152.2763L90.82195 152.068L91.3562 151.8582L91.89044 151.64554L92.4247 151.4286L92.95894 151.21593L93.49318 151.0004L94.02744 150.78912L94.56168 150.57358L95.09592 150.3566L95.63017 150.14104L96.16441 149.92409L96.69865 149.70709L97.23291 149.49294L97.76715 149.27599L98.30139 149.0604L98.83565 148.8434L99.36989 148.62642L99.90413 148.40944L100.43839 148.19524L100.97263 147.97823L101.50687 147.76122L102.04112 147.5442L102.57536 147.3272L103.1096 147.11018L103.64386 146.89314L104.1781 146.67609L104.71236 146.45903L105.2466 146.2461L105.78084 146.02899L106.3151 145.81184L106.84933 145.59468L107.38358 145.37753L107.91783 145.16031L108.45207 144.94305L108.98631 144.72568L109.52057 144.50827L110.05481 144.29082L110.58905 144.07323L111.1233 143.85555L111.65755 143.63766L112.19179 143.41965L112.72604 143.20145L113.26028 142.98297L113.79453 142.76419L114.32878 142.54507L114.86302 142.32553L115.39728 142.1055L115.93152 141.88484L116.46576 141.66345L117.00002 141.44118L117.53426 141.21788L118.0685 140.99335L118.60275 140.76726L119.137 140.53941L119.67123 140.30937L120.20549 140.07678L120.73973 139.84113L121.27397 139.60168L121.80823 139.3579L122.34247 139.10875L122.87671 138.85326L123.41096 138.59012L123.9452 138.31792L124.47945 138.03477L125.0137 137.73866L125.54794 137.42697L126.08218 137.09663L126.61644 136.74416L127.15068 136.36513L127.68494 135.95456L128.21918 135.50626L128.75342 135.01341L129.28767 134.46696L129.82191 133.85762L130.35616 133.17415L130.89041 132.40234L131.42465 131.52763L131.9589 130.53174L132.49315 129.39406L133.02739 128.09235L133.56163 126.60208L134.09589 124.8948L134.63013 122.94263L135.16437 120.71289L135.69862 118.17886L136.23286 115.30891L136.7671 112.0862L137.30136 108.48523L137.8356 104.50061L138.36984 100.13438L138.9041 95.40477L139.43834 90.34949L139.9726 85.01839L140.50684 79.4806L141.04108 73.81586L141.57533 68.11928L142.10957 62.47305L142.64381 56.96487L143.17807 51.68916L143.71231 46.70438L144.24655 42.06491L144.7808 37.80537L145.31505 33.93828L145.84929 30.47737L146.38354 27.40962L146.91779 24.7041L147.45203 22.34935L147.98628 20.30511L148.52052 18.5566L149.05476 17.05563L149.58902 15.7747L150.12326 14.68881L150.6575 13.76959L151.19176 12.99126L151.726 12.3353L152.26025 11.78802L152.7945 11.33005L153.32874 10.942L153.86299 10.62115L154.39723 10.35075L154.93147 10.12347L155.46573 9.93443L155.99997 9.77873L156.53421 9.64807L157.06847 9.53996L157.6027 9.44963L158.13695 9.37357L158.6712 9.31047L159.20544 9.25798L159.73969 9.21721L160.27394 9.17886L160.80818 9.14871L161.34242 9.12553L161.87668 9.10246L162.41092 9.08629L162.94516 9.07248L163.47942 9.05984L164.01366 9.04951L164.54791 9.04263L165.08215 9.03804L165.6164 9.02893L166.15065 9.02327L166.68489 9.01984L167.21913 9.01643L167.75339 9.01529L168.28763 9.0108L168.82187 9.00966L169.35612 9.00853L169.89037 9.00851L170.4246 9.00629L170.95886 9.00627L171.4931 9.00514L172.02734 9.00513H172.5616L173.09584 9.00291H173.63008L174.16434 9.0029H174.69858H175.23283L175.76707 9.00288L176.30132 8.99963H176.83557H177.36981H177.90405H178.43831H178.97255H179.50679H180.04105H180.57529H181.10953H181.64378H182.17802H182.71227H183.24652H183.78076H184.315H184.84926H185.3835H185.91774H186.452H186.98624H187.5205H188.05473H188.58897H189.12323H189.65747H190.19171H190.72597H191.26021H191.79445H192.3287H192.86295H193.39719H193.93144H194.46568H194.99992' fill='none'/>
</g>
<g stroke-dasharray='3.0,3.0' stroke-dashoffset='0.0'>
<g stroke-dasharray='2.0,2.0' stroke-dashoffset='0.0'>
<path d='M0 8.99963H194.99512H178.42053' fill='none'/>
<g transform='translate(92.47607,14.03261)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f1' x='11.175849' y='229.302819'>m</text>
<text class='f2' x='17.29306' y='230.797215'>1</text>
</g>
</g>
</g>
</g>
</g>
</g>
<g stroke-dasharray='3.0,3.0' stroke-dashoffset='0.0'>
<g stroke-dasharray='2.0,2.0' stroke-dashoffset='0.0'>
<path d='M58.49854 0V179.99268' fill='none'/>
<g transform='translate(62.03151,183.52565)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'/>
</g>
</g>
</g>
</g>
</g>
<g stroke-dasharray='3.0,3.0' stroke-dashoffset='0.0'>
<g stroke-dasharray='2.0,2.0' stroke-dashoffset='0.0'>
<path d='M141.37146 0V179.99268' fill='none'/>
<g transform='translate(144.90443,183.52565)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'/>
</g>
</g>
</g>
</g>
</g>
<path d='M53.92365 80.9967H63.07343' fill='none'/>
<g transform='matrix(-1.0,0.0,0.0,-1.0,53.82365,80.9967)'>
<g stroke-dasharray='none' stroke-dashoffset='0.0'>
 <g stroke-linecap='butt'>
 <g stroke-linejoin='miter'>
 <path d='M0 2.29999V-2.29999' fill='none'/>
 </g>
 </g>
 </g>
</g>
<g transform='translate(63.17342,80.9967)'>
<g stroke-dasharray='none' stroke-dashoffset='0.0'>
 <g stroke-linecap='butt'>
 <g stroke-linejoin='miter'>
 <path d='M0 2.29999V-2.29999' fill='none'/>
 </g>
 </g>
 </g>
</g>
<g transform='translate(39.79747,80.67032)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f1' x='11.175849' y='229.302819'>m</text>
<text class='f2' x='17.29306' y='230.797215'>4</text>
</g>
</g>
</g>
</g>
<path d='M145.94635 80.9967H136.79657' fill='none'/>
<g transform='translate(146.04634,80.9967)'>
<g stroke-dasharray='none' stroke-dashoffset='0.0'>
 <g stroke-linecap='butt'>
 <g stroke-linejoin='miter'>
 <path d='M0 2.29999V-2.29999' fill='none'/>
 </g>
 </g>
 </g>
</g>
<g transform='matrix(-1.0,0.0,0.0,-1.0,136.69658,80.9967)'>
<g stroke-dasharray='none' stroke-dashoffset='0.0'>
 <g stroke-linecap='butt'>
 <g stroke-linejoin='miter'>
 <path d='M0 2.29999V-2.29999' fill='none'/>
 </g>
 </g>
 </g>
</g>
<g transform='translate(150.02956,80.67032)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f1' x='11.175849' y='229.302819'>m</text>
<text class='f2' x='17.29306' y='230.797215'>6</text>
</g>
</g>
</g>
</g>
<path d='M141.37146 26.9989' fill='none'/>
<g transform='translate(127.79552,26.67252)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f1' x='11.175849' y='229.302819'>m</text>
<text class='f2' x='17.29306' y='230.797215'>5</text>
</g>
</g>
</g>
</g>
<path d='M58.49854 26.9989' fill='none'/>
<g transform='translate(62.03151,26.67252)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f1' x='11.175849' y='229.302819'>m</text>
<text class='f2' x='17.29306' y='230.797215'>3</text>
</g>
</g>
</g>
</g>
<g stroke-dasharray='3.0,3.0' stroke-dashoffset='0.0'>
<g stroke-dasharray='2.0,2.0' stroke-dashoffset='0.0'>
<path d='M68.24829 161.99341L194.99512 107.9956' fill='none'/>
<g transform='matrix(0.91997,-0.39192,0.39192,0.91997,119.7735,145.51254)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f1' x='11.175849' y='229.302819'>m</text>
<text class='f2' x='17.29306' y='230.797215'>1<tspan x='22.811003' y='229.302819'>+</tspan></text>
<text class='f1' x='29.560475' y='229.302819'>m</text>
<text class='f2' x='35.677686' y='230.797215'>2</text>
<text class='f0' x='41.195629' y='229.302819'>−</text>
<text class='f1' x='48.22183' y='229.302819'>m</text>
<text class='f2' x='54.339042' y='230.797215'>7</text>
<text class='f1' x='58.227293' y='229.302819'>t</text>
</g>
</g>
</g>
</g>
</g>
</g>
</g>
<g transform='translate(85.05772,-17.37112)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f1' x='11.175849' y='229.302819'>t</text>
<text class='f2' x='13.983646' y='229.302819'>(</text>
<text class='f1' x='16.681901' y='229.302819'>D<tspan x='22.383513'>O</tspan><tspan x='27.774476'>Y</tspan></text>
<text class='f2' x='33.264385' y='229.302819'>)</text>
</g>
</g>
</g>
</g>
<g transform='matrix(0.0,1.0,-1.0,0.0,-20.74628,69.74199)'>
<g stroke='none' transform='scale(-1.00375,1.00375)translate(11.175849,229.302819)scale(-1,-1)'>
<g fill='#000'>
<g stroke='none'>
<text class='f1' x='11.175849' y='229.302819'>v</text>
<text class='f2' x='14.952229' y='229.302819'>(</text>
<text class='f1' x='17.650484' y='229.302819'>t,<tspan x='23.963677'>M</tspan></text>
<text class='f2' x='30.903001' y='229.302819'>)</text>
</g>
</g>
</g>
</g>
</g>
</g>
</g>
</g>
</g>
</g>
</g>
</g>
</g>
</svg>" /></fr:resource-content><fr:resource-source
type="latex"
part="preamble"><![CDATA[
   
  \RequirePackage{tikz}
 \RequirePackage{amsmath}
 \usetikzlibrary{backgrounds, intersections, calc, spath3, fit}

 \definecolor{catccolor}{RGB}{255,244,138}
 \tikzstyle{dot}=[circle, draw=black, fill=black, minimum size=1mm, inner sep=0mm]
 \tikzstyle{catc}=[catccolor!60]
 \tikzstyle{catd}=[orange!40]
 \tikzstyle{cate}=[red!40]
 \tikzstyle{catf}=[blue!10]
 \tikzstyle{catg}=[green!25]


 \tikzstyle{blue halo}=[fill=blue!10, opacity=0.7, rounded corners]
 \tikzstyle{white halo}=[fill=white, opacity=0.7, rounded corners]

 \NewDocumentCommand\CreateRect{D<>{} m m}{
   \path
     coordinate (#1nw)
     ++(#2,-#3) coordinate (#1se)
     coordinate (#1sw) at (#1se -| #1nw)
     coordinate (#1ne) at (#1nw -| #1se)
   ;

   \path[spath/save = #1north] (#1nw) to (#1ne);
   \path[spath/save = #1west] (#1nw) to (#1sw);
   \path[spath/save = #1east] (#1ne) to (#1se);
   \path[spath/save = #1south] (#1sw) to (#1se);
 }

   
  \usepackage{tikz, tikz-cd, mathtools, amssymb, stmaryrd}

\usepackage{pgfplots}
\usepackage{amsmath}


 \usetikzlibrary{matrix,arrows, patterns}
 \usetikzlibrary{backgrounds,fit,positioning,calc,shapes}
 \usetikzlibrary{decorations.pathreplacing}
 \usetikzlibrary{decorations.pathmorphing}
 \usetikzlibrary{decorations.markings}

 \tikzset{
   desc/.style={sloped, fill=white,inner sep=2pt},
   upright desc/.style={fill=white,inner sep=2pt},
   pullback/.style = {
     append after command={
       \pgfextra{
         \draw ($(\tikzlastnode) + (.2cm,-.5cm)$) -- ++(0.3cm,0) -- ++(0,0.3cm);
       }
     }
   },
   pullback 45/.style = {
     append after command={
       \pgfextra{
         \draw[rotate = 45] ($(\tikzlastnode) + (.2cm,-.5cm)$) -- ++(0.3cm,0) -- ++(0,0.3cm);
       }
     }
   },
   ne pullback/.style = {
     append after command={
       \pgfextra{
         \draw ($(\tikzlastnode) + (-.2cm,-.5cm)$) -- ++(-0.3cm,0) -- ++(0,0.3cm);
       }
     }
   },
   sw pullback/.style = {
     append after command={
       \pgfextra{
         \draw ($(\tikzlastnode) + (.2cm,.5cm)$) -- ++(0.3cm,0) -- ++(0,-0.3cm);
       }
     }
   },
   dotted pullback/.style = {
     append after command={
       \pgfextra{
         \draw [densely dotted] ($(\tikzlastnode) + (.2cm,-.5cm)$) -- ++(0.3cm,0) -- ++(0,0.3cm);
       }
     }
   },
   muted pullback/.style = {
     append after command={
       \pgfextra{
         \draw [gray] ($(\tikzlastnode) + (.2cm,-.5cm)$) -- ++(0.3cm,0) -- ++(0,0.3cm);
       }
     }
   },
   pushout/.style = {
     append after command={
       \pgfextra{
         \draw ($(\tikzlastnode) + (-.2cm,.5cm)$) -- ++(-0.3cm,0) -- ++(0,-0.3cm);
       }
     }
   },
   between/.style args={#1 and #2}{
     at = ($(#1)!0.5!(#2)$)
   },
   diagram/.style = {
     on grid,
     node distance=2cm,
     commutative diagrams/every diagram,
     line width = .5pt,
     every node/.append style = {
       commutative diagrams/every cell,
     }
   },
   fibration/.style = {
     -{Triangle[open]}
   },
   etale/.style = {
     -{Triangle[open]}
   },
   etale cover/.style= {
     >={Triangle[open]},->.>
   },
   opfibration/.style = {
     -{Triangle}
   },
   lies over/.style = {
     |-{Triangle[open]}
   },
   op lies over/.style = {
     |-{Triangle}
   },
   embedding/.style = {
     {right hook}->
   },
   open immersion/.style = {
     {right hook}-{Triangle[open]}
   },
   closed immersion/.style = {
     {right hook}-{Triangle}
   },
   closed immersion*/.style = {
     {left hook}-{Triangle}
   },
   embedding*/.style = {
     {left hook}->
   },
   open immersion*/.style = {
     {left hook}-{Triangle[open]}
   },
   exists/.style = {
     densely dashed
   },
 }

 \newlength{\dontworryaboutit}

 \tikzset{
   inline diagram/.style = {
     commutative diagrams/every diagram,
     commutative diagrams/cramped,
     line width = .5pt,
     every node/.append style = {
       commutative diagrams/every cell,
       anchor = base,
       inner sep = 0pt
     },
     every path/.append style = {
       outer xsep = 2pt
     }
   }
 }

 \tikzset{
   square/nw/.style = {},
   square/ne/.style = {},
   square/se/.style = {},
   square/sw/.style = {},
   square/north/.style = {->},
   square/south/.style = {->},
   square/west/.style = {->},
   square/east/.style = {->},
   square/north/node/.style = {above},
   square/south/node/.style = {below},
   square/west/node/.style = {left},
   square/east/node/.style = {right},
 }

 \ExplSyntaxOn

 \bool_new:N \l_jon_glue_west

 \keys_define:nn { jon-tikz/diagram } {
   nw .tl_set:N = \l_jon_tikz_diagram_nw,
   sw .tl_set:N = \l_jon_tikz_diagram_sw,
   ne .tl_set:N = \l_jon_tikz_diagram_ne,
   se .tl_set:N = \l_jon_tikz_diagram_se,

   width .tl_set:N = \l_jon_tikz_diagram_width,
   height .tl_set:N = \l_jon_tikz_diagram_height,

   north .tl_set:N = \l_jon_tikz_diagram_north,
   south .tl_set:N = \l_jon_tikz_diagram_south,
   west .tl_set:N = \l_jon_tikz_diagram_west,
   east .tl_set:N = \l_jon_tikz_diagram_east,

   nw/style .code:n = {\tikzset{square/nw/.style = {#1}}},
   sw/style .code:n = {\tikzset{square/sw/.style = {#1}}},
   ne/style .code:n = {\tikzset{square/ne/.style = {#1}}},
   se/style .code:n = {\tikzset{square/se/.style = {#1}}},

   glue .choice:,
   glue / west .code:n = {\bool_set:Nn \l_jon_glue_west \c_true_bool},

   glue~target .tl_set:N = \l_jon_tikz_glue_target,

   north/style .code:n = {\tikzset{square/north/.style = {#1}}},
   north/node/style .code:n = {\tikzset{square/north/node/.style = {#1}}},
   south/style .code:n = {\tikzset{square/south/.style = {#1}}},
   south/node/style .code:n = {\tikzset{square/south/node/.style = {#1}}},
   west/style .code:n = {\tikzset{square/west/.style = {#1}}},
   west/node/style .code:n = {\tikzset{square/west/node/.style = {#1}}},
   east/style .code:n = {\tikzset{square/east/.style = {#1}}},
   east/node/style .code:n = {\tikzset{square/east/node/.style = {#1}}},

   draft .meta:n = {
     nw = {\__jon_tikz_diagram_fmt_placeholder:n {nw}},
     sw = {\__jon_tikz_diagram_fmt_placeholder:n {sw}},
     se = {\__jon_tikz_diagram_fmt_placeholder:n {se}},
     ne = {\__jon_tikz_diagram_fmt_placeholder:n {ne}},
     north = {\__jon_tikz_diagram_fmt_placeholder:n {north}},
     south = {\__jon_tikz_diagram_fmt_placeholder:n {south}},
     west = {\__jon_tikz_diagram_fmt_placeholder:n {west}},
     east = {\__jon_tikz_diagram_fmt_placeholder:n {east}},
   }
 }

 \tl_set:Nn \l_jon_tikz_diagram_width { 2cm }
 \tl_set:Nn \l_jon_tikz_diagram_height { 2cm }


 \cs_new:Npn \__jon_tikz_diagram_fmt_placeholder:n #1 {
   \texttt{\textcolor{red}{#1}}
 }

 \keys_set:nn { jon-tikz/diagram } {
   glue~target = {},
 }


 \cs_new:Nn \__jon_tikz_render_square:nn {
   \group_begin:
   \keys_set:nn {jon-tikz/diagram} {#2}
   \bool_if:nTF \l_jon_glue_west {
     \node (#1ne) [right = \l_jon_tikz_diagram_width~of~\l_jon_tikz_glue_target ne,square/ne] {$\l_jon_tikz_diagram_ne$};
     \node (#1se) [below = \l_jon_tikz_diagram_height~of~#1ne,square/se] {$\l_jon_tikz_diagram_se$};
     \draw[square/north] (\l_jon_tikz_glue_target ne) to node [square/north/node] {$\l_jon_tikz_diagram_north$} (#1ne);
     \draw[square/east] (#1ne) to node [square/east/node] {$\l_jon_tikz_diagram_east$} (#1se);
     \draw[square/south] (\l_jon_tikz_glue_target se) to node [square/south/node] {$\l_jon_tikz_diagram_south$} (#1se);
   } {
     \node (#1nw) [square/nw] {$\l_jon_tikz_diagram_nw$};
     \node (#1sw) [below = \l_jon_tikz_diagram_height~of~#1nw,square/sw] {$\l_jon_tikz_diagram_sw$};
     \draw[square/west] (#1nw) to node [square/west/node] {$\l_jon_tikz_diagram_west$} (#1sw);

     \node (#1ne) [right = \l_jon_tikz_diagram_width~of~#1nw,square/ne] {$\l_jon_tikz_diagram_ne$};
     \node (#1se) [below = \l_jon_tikz_diagram_height~of~#1ne,square/se] {$\l_jon_tikz_diagram_se$};
     \draw[square/north] (#1nw) to node [square/north/node] {$\l_jon_tikz_diagram_north$} (#1ne);
     \draw[square/east] (#1ne) to node [square/east/node] {$\l_jon_tikz_diagram_east$} (#1se);
     \draw[square/south] (#1sw) to node [square/south/node] {$\l_jon_tikz_diagram_south$} (#1se);
   }
   \group_end:
 }

 \NewDocumentCommand\SpliceDiagramSquare{D<>{}m}{
   \__jon_tikz_render_square:nn {#1} {#2}
 }


 \NewDocumentCommand\DiagramSquare{D<>{}O{}m}{
   \begin{tikzpicture}[diagram,#2,baseline=(#1sw.base)]
     \__jon_tikz_render_square:nn {#1} {#3}
   \end{tikzpicture}
 }

 \ExplSyntaxOff

  ]]></fr:resource-source><fr:resource-source
type="latex"
part="body"><![CDATA[
\begin {tikzpicture}[scale=2]
\tikzstyle {every node}=[font=\tiny ]
    \begin {axis}[
        domain=0:400,
        samples=366,
        xmin=0, xmax=400,
        ymin=0, ymax=0.9,
        xlabel={$t(DOY)$},
        ylabel={$v(t, M)$},
        xtick={0, 100, 200, 300},
        ytick={0, 0.2, 0.4, 0.6, 0.8},
        xlabel near ticks,
        ylabel near ticks,
        legend pos=north west,
       xtick pos=bottom,ytick pos=left
    ]

\pgfmathsetmacro {\mone }{0.05}    \pgfmathsetmacro {\mtwo }{1.00}    \pgfmathsetmacro {\mthree }{120}  \pgfmathsetmacro {\mfour }{8}    \pgfmathsetmacro {\mfive }{290}   \pgfmathsetmacro {\msix }{6}     \pgfmathsetmacro {\mseven }{0.0011} 
\addplot [black]
{(\mone  + (\mtwo  - \mseven  * x)*
 (1 / (1 + exp((\mthree  - x) / \mfour )) -
  1 / (1 + exp((\mfive  - x) / \msix ))))};


\draw [dashed, dash pattern=on 2pt off 2pt] (axis cs:0,\mone ) -- node[above] {$m_1$} (axis cs:400, \mone ) -- (axis cs:366,\mone ) ;
\draw [dashed, dash pattern=on 2pt off 2pt] (axis cs:\mthree ,0) -- (axis cs:\mthree ,1) node[above right] {};
\draw [dashed, dash pattern=on 2pt off 2pt] (axis cs:\mfive ,0) -- (axis cs:\mfive ,1) node[above right] {};

\draw [{| - |}] (axis cs:\mthree -10,0.45) -- (axis cs:\mthree +10,0.45) node[left, xshift=-10] {$m_4$};
\draw [{| - |}] (axis cs:\mfive +10,0.45) -- (axis cs:\mfive -10,0.45) node[right, xshift=10] {$m_6$};


\draw  (axis cs:\mfive ,0.15) node[left] {$m_5$};
\draw  (axis cs:\mthree ,0.15) node[right] {$m_3$};

\draw [dashed, dash pattern=on 2pt off 2pt] (axis cs:140,0.9) -- (axis cs:400,0.6)  node[midway,sloped,above, xshift=10] {$m_1 + m_2 - m_7 t$};

\end {axis}

\end {tikzpicture}

]]></fr:resource-source></fr:resource></fr:figure>
 
</fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="false"
numbered="false"
show-heading="true"
show-metadata="true"
expanded="false"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>250</fr:anchor><fr:addr
type="user">ocs-0003</fr:addr><fr:route>ocs-0003.xml</fr:route><fr:title
text="Arena Allocator">Arena Allocator</fr:title><fr:taxon>Note</fr:taxon><fr:date><fr:year>2024</fr:year><fr:month>5</fr:month><fr:day>29</fr:day></fr:date><fr:authors /></fr:frontmatter><fr:mainmatter><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>251</fr:anchor><fr:addr
type="user">ocs-0004</fr:addr><fr:route>ocs-0004.xml</fr:route><fr:title
text="Logic behind the allocator">Logic behind the allocator</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Only requires an offset or pointer to state the end of the last allocation. To allocate some memory from the arena it is as simple as moving the offset (or pointer) forward. The allocation has the complexity of <fr:tex
display="inline"><![CDATA[O(1)]]></fr:tex>.

 Due to being the simplest allocator possible, the arena allocator does nto allow the user to free certain blocks of memory. The memory is usually <fr:em>freed all at once</fr:em>.</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>252</fr:anchor><fr:addr
type="user">ocs-0005</fr:addr><fr:route>ocs-0005.xml</fr:route><fr:title
text="Basic arena interface">Basic arena interface</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>An arena is like a stack with extensions, e.g. it can be named.</fr:p><fr:pre><![CDATA[Arena *arena_alloc(void);
void arena_release(Arena *arena);
void arena_set_auto_align(Arena *arena, u64 align);

u64 arena_pos(Arena *arena);

void *arena_push_no_zero(Arena *arena, u64 size);
void *arena_push_aligner(Arena *arena, u64 alignment);
void *arena_push(Arena *arena, u64 size);

void arena_pop_to(Arena *arena, u64 pos);
void arena_pop(Arena *arena, u64 size);
void arena_clear(Arena *arena);

#define PushArrayNoZero(arena, type, count) (type *) \
    arena_push_no_zero((arena), sizeof(type) * (count))
#define PushArray(arena, type, count) (type *) \
    arena_push((arena), sizeof(type) * (count))]]></fr:pre><fr:p>the simplest arena allocator will look something like</fr:p><fr:pre><![CDATA[static unsigned char *arena_buffer;
static size_t arena_buffer_length;
static size_t arena_offset;

void *arena_alloc(size_t size) {
    // Check to see if the backing memory has space left
    if (arena_offset+size <= arena_buffer_length) {
        void *ptr = &arena_buffer[arena_offset];
        arena_offset += size;

        // Zero memory
        memset(ptr, 0, size);
        return ptr;
    }
    return NULL;
}]]></fr:pre><fr:p>While this is as simple as it gets, there are two issues with the basic approach
  * You cannot reuse this procedure for different arenas.
  * The pointer returned may not be aligned correctly for the data you need.

 The first issue can be solved by coupling the data into a structure and passing that to the procedure =arena_alloc=.</fr:p></fr:mainmatter><fr:backmatter /></fr:tree></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="false"
numbered="false"
show-heading="true"
show-metadata="true"
expanded="false"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>253</fr:anchor><fr:addr
type="user">ocs-0008</fr:addr><fr:route>ocs-0008.xml</fr:route><fr:title
text="Bash color codes">Bash color codes</fr:title><fr:taxon>Snippet</fr:taxon><fr:authors /></fr:frontmatter><fr:mainmatter><fr:pre><![CDATA[# Reset
Color_Off='\033[0m'       # Text Reset

# Regular Colors
Black='\033[0;30m'        # Black
Red='\033[0;31m'          # Red
Green='\033[0;32m'        # Green
Yellow='\033[0;33m'       # Yellow
Blue='\033[0;34m'         # Blue
Purple='\033[0;35m'       # Purple
Cyan='\033[0;36m'         # Cyan
White='\033[0;37m'        # White

# Bold
BBlack='\033[1;30m'       # Black
BRed='\033[1;31m'         # Red
BGreen='\033[1;32m'       # Green
BYellow='\033[1;33m'      # Yellow
BBlue='\033[1;34m'        # Blue
BPurple='\033[1;35m'      # Purple
BCyan='\033[1;36m'        # Cyan
BWhite='\033[1;37m'       # White

# Underline
UBlack='\033[4;30m'       # Black
URed='\033[4;31m'         # Red
UGreen='\033[4;32m'       # Green
UYellow='\033[4;33m'      # Yellow
UBlue='\033[4;34m'        # Blue
UPurple='\033[4;35m'      # Purple
UCyan='\033[4;36m'        # Cyan
UWhite='\033[4;37m'       # White

# Background
On_Black='\033[40m'       # Black
On_Red='\033[41m'         # Red
On_Green='\033[42m'       # Green
On_Yellow='\033[43m'      # Yellow
On_Blue='\033[44m'        # Blue
On_Purple='\033[45m'      # Purple
On_Cyan='\033[46m'        # Cyan
On_White='\033[47m'       # White

# High Intensity
IBlack='\033[0;90m'       # Black
IRed='\033[0;91m'         # Red
IGreen='\033[0;92m'       # Green
IYellow='\033[0;93m'      # Yellow
IBlue='\033[0;94m'        # Blue
IPurple='\033[0;95m'      # Purple
ICyan='\033[0;96m'        # Cyan
IWhite='\033[0;97m'       # White

# Bold High Intensity
BIBlack='\033[1;90m'      # Black
BIRed='\033[1;91m'        # Red
BIGreen='\033[1;92m'      # Green
BIYellow='\033[1;93m'     # Yellow
BIBlue='\033[1;94m'       # Blue
BIPurple='\033[1;95m'     # Purple
BICyan='\033[1;96m'       # Cyan
BIWhite='\033[1;97m'      # White

# High Intensity backgrounds
On_IBlack='\033[0;100m'   # Black
On_IRed='\033[0;101m'     # Red
On_IGreen='\033[0;102m'   # Green
On_IYellow='\033[0;103m'  # Yellow
On_IBlue='\033[0;104m'    # Blue
On_IPurple='\033[0;105m'  # Purple
On_ICyan='\033[0;106m'    # Cyan
On_IWhite='\033[0;107m'   # White]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="false"
numbered="false"
show-heading="true"
show-metadata="true"
expanded="false"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>254</fr:anchor><fr:addr
type="user">ocs-0009</fr:addr><fr:route>ocs-0009.xml</fr:route><fr:title
text="Single Instruction Multiple Vector (SIMD)">Single Instruction Multiple Vector (SIMD)</fr:title><fr:taxon>Note</fr:taxon><fr:date><fr:year>2025</fr:year><fr:month>3</fr:month><fr:day>20</fr:day></fr:date><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Single vector instruction sets are extensions to the original architecture instruction sets, and thus you have to invest extra effort to use and support them. With compiled code, you need to write inline assembly, or use vector intrinsics.</fr:p><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>255</fr:anchor><fr:addr
type="user">ocs-000J</fr:addr><fr:route>ocs-000J.xml</fr:route><fr:title
text="Vector intrinsics">Vector intrinsics</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Intrinsics appear like regular libary functions. You include the relevant header, and then you can use it just like any other lib. E.g. assuming a visual c++ compiler, if you wanted to add four float numbers to another four numbers, you would include the <fr:code>xmmintrin.h</fr:code> header which contains the following declaration for the <fr:code>_mm_add_ps</fr:code> intrinsic:</fr:p><fr:pre><![CDATA[extern _m128 _mm_add_ps( _m128 _A, _m128 _B);]]></fr:pre><fr:p>However unlike lib functions, intrinsics are implemented directly in compilers. The example SSE intrinsic typically compiles to a single instruction, x86 and amd64 instructions. For the time it takes CPU to call a library function, it might have completed dozen of these functions.</fr:p><fr:p>For all x86 the current best practice is to use <fr:code>&lt;immintrin.h&gt;</fr:code> which loads in the the correct compatible instruction set.
<fr:ul><fr:li>__MMX__ -- X86 MMX :: <fr:code>mmintrin.h</fr:code></fr:li>
<fr:li>__SSE__ -- X86 SSE :: <fr:code>xmmintrin.h</fr:code></fr:li>
<fr:li>__SSE2__ -- X86 SSE2 :: <fr:code>emmintrin.h</fr:code></fr:li>
<fr:li>__VEC__ -- altivec functions :: <fr:code>altivec.h</fr:code></fr:li>
<fr:li>__ARM_NEON__ -- ARM Neon functions :: <fr:code>arm_neon.h</fr:code></fr:li></fr:ul></fr:p>
  <html:table
xmlns:html="http://www.w3.org/1999/xhtml">
  
  
  <html:tr>
    
  <html:td><fr:strong>Operation</fr:strong></html:td>

    
  <html:td><fr:strong>Altivec</fr:strong></html:td>

    
  <html:td><fr:strong>Neon</fr:strong></html:td>

    
  <html:td><fr:strong>MMX/SSE/SSE2</fr:strong></html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>
  <html:hr />
</html:td>

    
  <html:td>
  <html:hr />
</html:td>

    
  <html:td>
  <html:hr />
</html:td>

    
  <html:td>
  <html:hr />
</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>loading</html:td>

    
  <html:td>vec_ld</html:td>

    
  <html:td>vld1q_f32</html:td>

    
  <html:td>_mm_set_epi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>vector</html:td>

    
  <html:td>vec_splat</html:td>

    
  <html:td>vld1q_s16</html:td>

    
  <html:td>_mm_set1_epi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_splat_s16</html:td>

    
  <html:td>vsetq_lane_f32</html:td>

    
  <html:td>_mm_set1_pi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_splat_s32</html:td>

    
  <html:td>vld1_u8</html:td>

    
  <html:td>_mm_set_pi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_splat_s8</html:td>

    
  <html:td>vdupq_lane_s16</html:td>

    
  <html:td>_mm_load_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_splat_u16</html:td>

    
  <html:td>vdupq_n_s16</html:td>

    
  <html:td>_mm_set1_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_splat_u32</html:td>

    
  <html:td>vmovq_n_f32</html:td>

    
  <html:td>_mm_loadh_pi</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_splat_u8</html:td>

    
  <html:td>vset_lane_u8</html:td>

    
  <html:td>_mm_loadl_pi</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>storing</html:td>

    
  <html:td>vec_st</html:td>

    
  <html:td>vst1_u8</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td>vector</html:td>

    
  <html:td />

    
  <html:td>vst1q_s16</html:td>

    
  <html:td>_mm_store_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vst1q_f32</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vst1_s16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td>add</html:td>

    
  <html:td>vec_madd</html:td>

    
  <html:td>vaddq_s16</html:td>

    
  <html:td>_mm_add_epi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_mladd</html:td>

    
  <html:td>vaddq_f32</html:td>

    
  <html:td>_mm_add_pi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_adds</html:td>

    
  <html:td>vmlaq_n_f32</html:td>

    
  <html:td>_mm_add_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>subtract</html:td>

    
  <html:td>vec_sub</html:td>

    
  <html:td>vsubq_s16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td>multiply</html:td>

    
  <html:td>vec_madd</html:td>

    
  <html:td>vmulq_n_s16</html:td>

    
  <html:td>_mm_mullo_epi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_mladd</html:td>

    
  <html:td>vmulq_s16</html:td>

    
  <html:td>_mm_mullo_pi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vmulq_f32</html:td>

    
  <html:td>_mm_mul_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vmlaq_n_f32</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td>arithmetic</html:td>

    
  <html:td>vec_sra</html:td>

    
  <html:td>vshrq_n_s16</html:td>

    
  <html:td>_mm_srai_epi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>shift</html:td>

    
  <html:td>vec_srl</html:td>

    
  <html:td />

    
  <html:td>_mm_srai_pi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_sr</html:td>

    
  <html:td />

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td>byte</html:td>

    
  <html:td>vec_perm</html:td>

    
  <html:td>vtbl1_u8</html:td>

    
  <html:td>_mm_shuffle_pi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>permutation</html:td>

    
  <html:td>vec_sel</html:td>

    
  <html:td>vtbx1_u8</html:td>

    
  <html:td>_mm_shuffle_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_mergeh</html:td>

    
  <html:td>vget_high_s16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_mergel</html:td>

    
  <html:td>vget_low_s16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vdupq_lane_s16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vdupq_n_s16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vmovq_n_f32</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vbsl_u8</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td>type</html:td>

    
  <html:td>vec_cts</html:td>

    
  <html:td>vmovl_u8</html:td>

    
  <html:td>_mm_packs_pu16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>conversion</html:td>

    
  <html:td>vec_unpackh</html:td>

    
  <html:td>vreinterpretq_s16_u16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_unpackl</html:td>

    
  <html:td>vcvtq_u32_f32</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_cts</html:td>

    
  <html:td>vqmovn_s32</html:td>

    
  <html:td>_mm_cvtps_pi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td>vec_ctu</html:td>

    
  <html:td>vqmovun_s16</html:td>

    
  <html:td>_mm_packus_epi16</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vqmovn_u16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vcvtq_f32_s32</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vmovl_s16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vmovq_n_f32</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td>vector</html:td>

    
  <html:td>vec_pack</html:td>

    
  <html:td>vcombine_u16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td>combination</html:td>

    
  <html:td>vec_packsu</html:td>

    
  <html:td>vcombine_u8</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td>vcombine_s16</html:td>

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td>maximum</html:td>

    
  <html:td />

    
  <html:td />

    
  <html:td>_mm_max_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>minimum</html:td>

    
  <html:td />

    
  <html:td />

    
  <html:td>_mm_min_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>vector</html:td>

    
  <html:td />

    
  <html:td />

    
  <html:td>_mm_andnot_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>logic</html:td>

    
  <html:td />

    
  <html:td />

    
  <html:td>_mm_and_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td />

    
  <html:td />

    
  <html:td />

    
  <html:td>_mm_or_ps</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>rounding</html:td>

    
  <html:td>vec_trunc</html:td>

    
  <html:td />

    
  <html:td />

  </html:tr>



  
  
  <html:tr>
    
  <html:td>misc</html:td>

    
  <html:td />

    
  <html:td />

    
  <html:td>_mm_empty</html:td>

  </html:tr>



  
  
  <html:tr>
    
  <html:td>
  <html:hr />
</html:td>

    
  <html:td>
  <html:hr />
</html:td>

    
  <html:td>
  <html:hr />
</html:td>

    
  <html:td>
  <html:hr />
</html:td>

  </html:tr>



</html:table>
</fr:mainmatter><fr:backmatter /></fr:tree><fr:p>Links
  <fr:ul><fr:li><fr:link
type="external"
href="https://www.youtube.com/watch?v=DXPfE2jGqg0">Utilizing the other 80% of your system's performance: Starting with Vectorization</fr:link></fr:li>
    <fr:li><fr:link
type="external"
href="https://stackoverflow.blog/2020/07/08/improving-performance-with-simd-intrinsics-in-three-use-cases/">Improving perf with simd intrinsics in three use cases</fr:link></fr:li>
    <fr:li><fr:link
type="external"
href="http://ftp.cvut.cz/kernel/people/geoff/cell/ps3-linux-docs/CellProgrammingTutorial/BasicsOfSIMDProgramming.html">Basics of SIMD programming</fr:link></fr:li>
    <fr:li><fr:link
type="external"
href="http://librestats.com/RparallelGuide/#the-simd-keyword">SIMD Keyword OpenMP, R, Parallelism</fr:link></fr:li>
    <fr:li><fr:link
type="external"
href="https://lowleveldev.substack.com/p/simd-a-practical-guide">SIMD a practical guide</fr:link></fr:li>
    <fr:li><fr:link
type="external"
href="https://stackoverflow.com/questions/70569824/organizing-multiple-implementations-for-simd">Organizing SIMD code - stackoverflow</fr:link></fr:li>
    <fr:li><fr:link
type="external"
href="https://stackoverflow.com/questions/661338/sse-sse2-and-sse3-for-gnu-c">SSE SSE2 and SSE3 for GNU C++ [closed] - stackoverflow</fr:link></fr:li>
    <fr:li><fr:link
type="external"
href="https://statmodeling.stat.columbia.edu/2023/11/01/simd-memory-locality-vectorization-and-branch-point-prediction/">SIMD, memory locality, vectorization, and branch point prediction</fr:link></fr:li>
    <fr:li><fr:link
type="external"
href="https://eclecticlight.co/2021/07/27/code-in-arm-assembly-rounding-and-arithmetic/">Code in ARM assembly: Rounding and arithmetic (scalar)</fr:link></fr:li>
    <fr:li><fr:link
type="external"
href="https://blog.yiningkarlli.com/2021/09/neon-vs-sse.html">Comparing SIMD on x86-64 and arm64</fr:link></fr:li></fr:ul></fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="false"
numbered="false"
show-heading="true"
show-metadata="true"
expanded="false"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>256</fr:anchor><fr:addr
type="user">ocs-000B</fr:addr><fr:route>ocs-000B.xml</fr:route><fr:title
text="Makefiles">Makefiles</fr:title><fr:taxon>Note</fr:taxon><fr:date><fr:year>2024</fr:year><fr:month>9</fr:month><fr:day>13</fr:day></fr:date><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>A well written makefile describes all the files and settings used to compile a project and link it with the appropriate libraries</fr:p><fr:p>Managing a build with a makefile is less error-prone and far simpler after a _one-time setup cost_. A makefile with a proper dependency graph will only rebuild a target when one or more of the components it depends on has changed. This allows seperating the components of a project whcih can be built independently.</fr:p><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>257</fr:anchor><fr:addr
type="user">ocs-000D</fr:addr><fr:route>ocs-000D.xml</fr:route><fr:title
text="Makefile syntax">Makefile syntax</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>It is common practice to copy and modify old makefules rather than generating new ones from scratch.</fr:p><fr:pre><![CDATA[#
# A simple makefile for managing build of project composed of C source files.
#


# It is likely that default C compiler is already gcc, but explicitly
# set, just to be sure
CC = gcc

# The CFLAGS variable sets compile flags for gcc:
#  -g        compile with debug information
#  -Wall     give verbose compiler warnings
#  -O0       do not optimize generated code
#  -std=c99  use the C99 standard language definition
CFLAGS = -g -Wall -O0 -std=c99

# The LDFLAGS variable sets flags for linker
#  -lm   says to link in libm (the math library)
LDFLAGS = -lm

# In this section, you list the files that are part of the project.
# If you add/change names of source files, here is where you
# edit the Makefile.
SOURCES = demo.c vector.c map.c
OBJECTS = $(SOURCES:.c=.o)
TARGET = demo


# The first target defined in the makefile is the one
# used when make is invoked with no argument. Given the definitions
# above, this Makefile file will build the one named TARGET and
# assume that it depends on all the named OBJECTS files.

$(TARGET) : $(OBJECTS)
    $(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS)

# Phony means not a "real" target, it doesn't build anything
# The phony target "clean" is used to remove all compiled object files.

.PHONY: clean

clean:
    @rm -f $(TARGET) $(OBJECTS) core]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>258</fr:anchor><fr:addr
type="user">ocs-000E</fr:addr><fr:route>ocs-000E.xml</fr:route><fr:title
text="Makefile macros">Makefile macros</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Subsitutions defined toward top of the file, e.g. (<fr:code>CFLAGS = -g -Wall</fr:code>), and are similar to <fr:code>#define</fr:code> in <fr:link
type="external"
href="%7B%7B%3C%20relref%20%2220240401132118-c.md%22%20%3E%7D%7D">C</fr:link> . It can be referenced later with <fr:code>$(CFLAGS)</fr:code>.</fr:p><fr:p>Example macros: - <fr:code>OBJECTS = $(SOURCES:.c=o)</fr:code> - Defines the <fr:code>OBJECTS</fr:code> macro to be the same as the <fr:code>SOURCES</fr:code> macro except that every instance of <fr:code>.c</fr:code> is replaced with <fr:code>.o</fr:code>. - <fr:code>$@</fr:code>: - built-in, name of the current target - <fr:code>$^</fr:code>: - built-in, current targets list of dependencies</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>259</fr:anchor><fr:addr
type="user">ocs-000F</fr:addr><fr:route>ocs-000F.xml</fr:route><fr:title
text="Targets">Targets</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Written in the following form</fr:p><fr:pre><![CDATA[target-name : dependencies
    action]]></fr:pre><fr:p>where the target name is the name of the file that will be produced when this target is built. The first target listed in a make file is the default target, meaning it is invoked by <fr:code>make</fr:code> with no args.</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>260</fr:anchor><fr:addr
type="user">ocs-000C</fr:addr><fr:route>ocs-000C.xml</fr:route><fr:title
text="Phony targets">Phony targets</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>The clean target in the example does not fit the pattern which is     previously described of targets. It does not create a file named     <fr:code>clean</fr:code>, rather it is a shortcut for running a command whcih cclears out     the projects build files. These commands are listed as dependencies of     <fr:code>.PHONY</fr:code> which allows Make to run the target without any dependency     checks.</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:p>links</fr:p><fr:ul><fr:li><fr:link
type="external"
href="https://web.stanford.edu/class/archive/cs/cs107/cs107.1174/guide_make.html">cs107 makefile guide</fr:link></fr:li>
<fr:li><fr:link
type="external"
href="https://gist.github.com/isaacs/62a2d1825d04437c6f08">Makefile basics - gist</fr:link></fr:li></fr:ul></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="false"
numbered="false"
show-heading="true"
show-metadata="true"
expanded="false"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>261</fr:anchor><fr:addr
type="user">go-00rs</fr:addr><fr:route>go-00rs.xml</fr:route><fr:title
text="Golang">Golang</fr:title><fr:taxon>Language</fr:taxon><fr:authors /></fr:frontmatter><fr:mainmatter><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>262</fr:anchor><fr:addr
type="user">go-00rv</fr:addr><fr:route>go-00rv.xml</fr:route><fr:title
text="Initialize module">Initialize module</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Within a new directory, initialize a module with</fr:p><fr:pre><![CDATA[go mod init your/module]]></fr:pre><fr:p>This will create a new file `go.mod`</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>263</fr:anchor><fr:addr
type="user">go-00rx</fr:addr><fr:route>go-00rx.xml</fr:route><fr:title
text="Types">Types</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter>
  <html:table
xmlns:html="http://www.w3.org/1999/xhtml">
  
  
  <html:tr>
    
  <html:td><fr:strong>Zero:</fr:strong></html:td>

    
  <html:td>Predeclared</html:td>

    
  <html:td>Default value assigned to declared but unassigned variables</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>Literals:</fr:strong></html:td>

    
  <html:td>Predeclared</html:td>

    
  <html:td>Explicitly specified values</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>Integer:</fr:strong></html:td>

    
  <html:td>Literals</html:td>

    
  <html:td>Base 10 by default</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>Floating-point:</fr:strong></html:td>

    
  <html:td>Literals</html:td>

    
  <html:td>Uses decimal point for fractions</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>Rune:</fr:strong></html:td>

    
  <html:td>Literals</html:td>

    
  <html:td>Single character representation</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>String:</fr:strong></html:td>

    
  <html:td>Literals</html:td>

    
  <html:td>Two creation methods</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>Booleans:</fr:strong></html:td>

    
  <html:td>Predeclared</html:td>

    
  <html:td>Binary true/false values</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>byte:</fr:strong></html:td>

    
  <html:td>Integer</html:td>

    
  <html:td>Alias for uint8</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>int:</fr:strong></html:td>

    
  <html:td>Integer</html:td>

    
  <html:td>Signed integer sized to architecture's word size</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>uint:</fr:strong></html:td>

    
  <html:td>Integer</html:td>

    
  <html:td>Unsigned integer sized to architecture's word size</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>rune:</fr:strong></html:td>

    
  <html:td>Integer</html:td>

    
  <html:td>Alias for int32</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>uintptr:</fr:strong></html:td>

    
  <html:td>Integer</html:td>

    
  <html:td>Integer type for storing pointer values</html:td>

  </html:tr>


  
  
  <html:tr>
    
  <html:td><fr:strong>String:</fr:strong></html:td>

    
  <html:td>Predeclared</html:td>

    
  <html:td>Immutable sequence of bytes, zero value is empty string</html:td>

  </html:tr>


</html:table>
</fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>264</fr:anchor><fr:addr
type="user">go-00ry</fr:addr><fr:route>go-00ry.xml</fr:route><fr:title
text="Type conversion">Type conversion</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Go does not support automatic type conversion. All variable types need to be converted to interact.</fr:p><fr:pre><![CDATA[var x int = 10
var y float64 = 30.2
var sum1 float64 = float64(x) + y
var sum2 int = x + int(y)
fmt.Println(sum1, sum2) // 40.2  40]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>265</fr:anchor><fr:addr
type="user">go-00rz</fr:addr><fr:route>go-00rz.xml</fr:route><fr:title
text="Quotes">Quotes</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Different qoutes are used for different contexts.</fr:p><fr:p>The double qoute <fr:code>"</fr:code> will be used to create string literals, while the backtick is used to create <fr:em>raw</fr:em> string literals. The single qoute <fr:code>'</fr:code> is used to create runes, a single unicode point. E.g. <fr:code>r := '學'</fr:code>.</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>266</fr:anchor><fr:addr
type="user">go-00s0</fr:addr><fr:route>go-00s0.xml</fr:route><fr:title
text="Slices">Slices</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:ul><fr:li>Slices will have a zero value of <fr:code>nil</fr:code> and is only comprable with <fr:code>nil</fr:code> and <fr:code>==</fr:code>.</fr:li>
    <fr:li>Modifying shared slices will affect all references.</fr:li>
    <fr:li>Length can differ from capacity</fr:li></fr:ul><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>267</fr:anchor><fr:addr
type="user">go-00s3</fr:addr><fr:route>go-00s3.xml</fr:route><fr:title
text="Creation">Creation</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:pre><![CDATA[var s []int             // nil slice
s := []int{1, 2, 3}     // slice literal
s := make([]int, 5)     // length 5, capacity 5
s := make([]int, 0, 10) // length 0, capacity 10]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>268</fr:anchor><fr:addr
type="user">go-00s4</fr:addr><fr:route>go-00s4.xml</fr:route><fr:title
text="Modification">Modification</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:pre><![CDATA[s = append(s, 10)           // single element
s = append(s, 1, 2, 3)      // multiple elements
s = append(s, other...)     // append slice
clear(s)                    // zero all elements (Go 1.21+)]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>269</fr:anchor><fr:addr
type="user">go-00s5</fr:addr><fr:route>go-00s5.xml</fr:route><fr:title
text="Slicing">Slicing</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:pre><![CDATA[s[low:high]     // items low through high-1
s[low:]         // items low through end
s[:high]        // items beginning through high-1
s[:]            // copy of entire slice
s[low:high:max] // control capacity of new slice]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>270</fr:anchor><fr:addr
type="user">go-00s6</fr:addr><fr:route>go-00s6.xml</fr:route><fr:title
text="Copying">Copying</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:pre><![CDATA[copy(dest, src)        // copies min(len(dest), len(src)) elements
copy(dest, src[1:])    // copy from middle of src]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>271</fr:anchor><fr:addr
type="user">go-00s7</fr:addr><fr:route>go-00s7.xml</fr:route><fr:title
text="Array Conversion">Array Conversion</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:pre><![CDATA[array := [4]int{1, 2, 3, 4}
slice := array[:]              // array to slice
array2 := [4]int(slice)       // slice to array (new memory)
arrayPtr := (*[4]int)(slice)  // slice to array pointer (shared memory)]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>272</fr:anchor><fr:addr
type="user">go-00s9</fr:addr><fr:route>go-00s9.xml</fr:route><fr:title
text="When to Use What">When to Use What</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>1.  `var` declaration (nil slice): When size unknown, no initial data 2.  Slice literal: When starting values are known 3.  `make`:     -   With length: For buffers or exact size known     -   Zero length + capacity: For growing slices</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>273</fr:anchor><fr:addr
type="user">go-00sa</fr:addr><fr:route>go-00sa.xml</fr:route><fr:title
text="Memory Management">Memory Management</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>-   Capacity doubles when &amp;lt; 256 -   Growth slows to ~25/for larger slices -   Use full slice expression `s[low:high:max]` to prevent capacity issues -   Be careful with shared memory when slicing</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>274</fr:anchor><fr:addr
type="user">go-00sb</fr:addr><fr:route>go-00sb.xml</fr:route><fr:title
text="Standard Library">Standard Library</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>-   `slices.Equal(a, b)`: Compare if length and elements match -   `slices.EqualFunc(a, b, fn)`: Custom comparison function</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>275</fr:anchor><fr:addr
type="user">go-00sc</fr:addr><fr:route>go-00sc.xml</fr:route><fr:title
text="Maps">Maps</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Maps are declared as `map[keyType]valueType`. A `nil` map (e.g., `var m map[string]int`) cannot be written to. Use `make` or a map literal for writable maps.</fr:p><fr:pre><![CDATA[// Map literal
m := map[string]int{"Alice": 10, "Bob": 20}

// Using make
m := make(map[string]int, 10) // Capacity 10
m["Charlie"] = 30]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>276</fr:anchor><fr:addr
type="user">go-00sd</fr:addr><fr:route>go-00sd.xml</fr:route><fr:title
text="Structs">Structs</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Structs are user-defined types with fields. Define them with `type` and access fields using dot notation.</fr:p><fr:pre><![CDATA[type Person struct {
    Name string
    Age  int
}
p := Person{Name: "Alice", Age: 30}
fmt.Println(p.Name)]]></fr:pre><fr:p>Anonymous structs are inline struct definitions:</fr:p><fr:pre><![CDATA[pet := struct {
    Name string
    Kind string
}{Name: "Fido", Kind: "dog"}]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>277</fr:anchor><fr:addr
type="user">go-00se</fr:addr><fr:route>go-00se.xml</fr:route><fr:title
text="Pointers">Pointers</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Use `&amp;` to get a pointer and `*` to dereference. `nil` pointers cannot be dereferenced.</fr:p><fr:pre><![CDATA[x := 10
p := &x
fmt.Println(*p) // 10]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>278</fr:anchor><fr:addr
type="user">go-00sf</fr:addr><fr:route>go-00sf.xml</fr:route><fr:title
text="Shadowing">Shadowing</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Inner-scope variables can shadow outer-scope ones:</fr:p><fr:pre><![CDATA[x := 10
if x > 5 {
    x := 5
    fmt.Println(x) // 5
}
fmt.Println(x) // 10]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>279</fr:anchor><fr:addr
type="user">go-00sg</fr:addr><fr:route>go-00sg.xml</fr:route><fr:title
text="Control Flow">Control Flow</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>280</fr:anchor><fr:addr
type="user">go-00sh</fr:addr><fr:route>go-00sh.xml</fr:route><fr:title
text="If">If</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Go's `if` statements do not require parentheses. You can declare variables scoped to the condition:</fr:p><fr:pre><![CDATA[if n := rand.Intn(10); n > 5 {
    fmt.Println("Big:", n)
} else {
    fmt.Println("Small:", n)
}]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>281</fr:anchor><fr:addr
type="user">go-00si</fr:addr><fr:route>go-00si.xml</fr:route><fr:title
text="For">For</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Loops in go support complete (C-style), condition only, for-range styles.</fr:p><fr:pre><![CDATA[// C-style
for i := 0; i < 10; i++ {
    fmt.Println(i)
}

// condition only (while)
i := 1
for i < 10 {
    fmt.Println(i)
    i++
}

// for-range
nums := []int{1, 2, 3}
for i, v := range nums {
    fmt.Println(i, v)
}]]></fr:pre><fr:p>If you don't need the key/index in,  the underscore <fr:code>_</fr:code> can be used, telling go to ignore the value.</fr:p><fr:pre><![CDATA[evenVals := []int{2, 4, 6, 8, 10, 12}
for _, v := range evenVals {
    fmt.Printf("%d ", v)
}]]></fr:pre><fr:p>If you just need the key/index, you can leave off the second variable:</fr:p><fr:pre><![CDATA[uniqueNames := map[string]bool{"Fred": true, "Raul": true, "Wilma": true}
for k := range uniqueNames {
    fmt.Printf("%s ", k)
}]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>282</fr:anchor><fr:addr
type="user">go-00sj</fr:addr><fr:route>go-00sj.xml</fr:route><fr:title
text="Switch">Switch</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:pre><![CDATA[words := []string{"a", "cow", "smile", "gopher",
    "octopus", "anthropologist"}
for _, word := range words {
    switch size := len(word); size {
    case 1, 2, 3, 4:
        fmt.Println(word, "is a short word!")
    case 5:
        wordLen := len(word)
        fmt.Println(word, "is exactly the right length:", wordLen)
    case 6, 7, 8, 9:
    default:
        fmt.Println(word, "is a long word!")
    }
}]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>283</fr:anchor><fr:addr
type="user">go-00sk</fr:addr><fr:route>go-00sk.xml</fr:route><fr:title
text="Functions">Functions</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>284</fr:anchor><fr:addr
type="user">go-00sl</fr:addr><fr:route>go-00sl.xml</fr:route><fr:title
text="Variadic Parameters">Variadic Parameters</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Variadic functions accept a variable number of arguments as a slice:</fr:p><fr:pre><![CDATA[func addTo(base int, vals ...int) []int {
    out := make([]int, 0, len(vals))
    for _, v := range vals {
        out = append(out, base+v)
    }
    return out
}]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>285</fr:anchor><fr:addr
type="user">go-00sm</fr:addr><fr:route>go-00sm.xml</fr:route><fr:title
text="Multiple Return Values">Multiple Return Values</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Functions can return multiple values, often used with errors:</fr:p><fr:pre><![CDATA[func divAndRemainder(num, denom int) (int, int, error) {
    if denom == 0 {
        return 0, 0, errors.New("cannot divide by zero")
    }
    return num / denom, num % denom, nil
}]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>286</fr:anchor><fr:addr
type="user">go-00sn</fr:addr><fr:route>go-00sn.xml</fr:route><fr:title
text="Named return values">Named return values</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Functions can have names specified for the return values, which predeclares variables that are used within the function to hold the return values. These values are initialized to their zero values allowing you to return them before any explicit use or assignment.</fr:p><fr:pre><![CDATA[func divAndRemainder(num, denom int) (result int, remainder int, err error) {
    if denom == 0 {
        err = errors.New("cannot divide by zero")
        return result, remainder, err
    }
    result, remainder = num / denom, num % denom
    return result, remainder, err
}]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>287</fr:anchor><fr:addr
type="user">go-00sp</fr:addr><fr:route>go-00sp.xml</fr:route><fr:title
text="Closures">Closures</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>These are functions defined _within_ a function, and are able to access and modify variables declared in the outer function. Closures allow you to pass fome function state to another function. These can also be returned from a function.</fr:p><fr:pre><![CDATA[package main

import "fmt"

func main() {
    a := 20
    f := func() {
        fmt.Println(a)
        a = 30
    }
    f()
    fmt.Println(a)
}]]></fr:pre><fr:p>Anonymous function `f` can read and write `a` even though `a` is not passed in to the function. Using `:=` instead of `=` inside the closure creates a new `a` that ceases to exist when the closure exits.</fr:p><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>288</fr:anchor><fr:addr
type="user">go-00sq</fr:addr><fr:route>go-00sq.xml</fr:route><fr:title
text="Why use closures?">Why use closures?</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:ul><fr:li>Limit function scope</fr:li>
<fr:p>If a function is going to be called from only one other function, but it's called multiple times, you can use an inner function to "hide" the called function. This reduces the number of declarations at a package level, making it easier to find an unused name.</fr:p>
<fr:li>Removing logic repetetion</fr:li>
<fr:p>Given a piece of logic that is repeated multiple times within a function, a closure can be used to remove that repetition.</fr:p>
<fr:li>Passing to funcs/returning from funcs</fr:li>
<fr:p>These become really interesting, allowing you to take variables within your function and use those variables <fr:em>outside</fr:em> of the function</fr:p></fr:ul></fr:mainmatter><fr:backmatter /></fr:tree></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>289</fr:anchor><fr:addr
type="user">go-00sv</fr:addr><fr:route>go-00sv.xml</fr:route><fr:title
text="Pointers">Pointers</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Standard address (`&amp;`) and indirection/dereferencing (`*`) operators. Pointer tricks such as _pointer arithmetic_ that you can do in C are not allowed in go.</fr:p><fr:p>Before dereferencing a pointer, you must make sure it is non-nil, otherwise the program will panic if you attempt to dereference a `nil` pointer:</fr:p><fr:pre><![CDATA[var * int
fmt.Println(x == nil)			// prints true
fmt.Println(*x)					// panics]]></fr:pre><fr:p>The pointer type can be based on any type, and the built in function `new` creates a pointer variable which returns a pointer to a zero-value instance of the type.</fr:p><fr:pre><![CDATA[var x = new(int)
fmt.Println(x == nil)			// prints false
fmt.Println(*x)					// prints 0]]></fr:pre><fr:p>However, the `new` function is rarely used.</fr:p><fr:p>For structs use `&amp;` before a struct literal to create a pointer instance. When you need a pointer to a primitive type, declare a variable and then point to it.</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>290</fr:anchor><fr:addr
type="user">go-00sw</fr:addr><fr:route>go-00sw.xml</fr:route><fr:title
text="Pointers of constants / literals">Pointers of constants / literals</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Constant literals cannot use `&amp;` because they don't have a memory address, e.g.</fr:p><fr:pre><![CDATA[type person struct {
    FirstName string
    MiddleName *string
    LastName string
}

p := person {
        FirstName: "Pat",
        MiddleName: "Perry", // This line will not compile
        LastName: "Peterson",
    }]]></fr:pre><fr:p>There are two ways around the problem. The first is to introduce a variable to hold the constant. The second is to write a generic helper function that takes a parameter of any type and returns a pointer to that type:</fr:p><fr:pre><![CDATA[func makePointer[T any](t T) *T {
  return &t
}]]></fr:pre><fr:p>which allows one to write the previous code as</fr:p><fr:pre><![CDATA[p := person {
        FirstName: "Pat",
        MiddleName: makePointer("Perry"),
        LastName: "Peterson",
    }]]></fr:pre><fr:p>**NOTE**: The only time you should use pointer parameters to modify a variable is when the function expects an interface, e.g. JSON.</fr:p></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="true"
numbered="true"
show-heading="true"
show-metadata="false"
expanded="true"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>291</fr:anchor><fr:addr
type="user">go-00sx</fr:addr><fr:route>go-00sx.xml</fr:route><fr:title
text="Difference between maps and slices">Difference between maps and slices</fr:title><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Within the go runtime, a map is implemented as a pointer to a struct, so passing a map to a function means that you are copying a pointer. Thus changes made to a map within the functions are reflected in the original.</fr:p><fr:p>**NOTE**: maps for input parameters or return values should be carefully considered. On an API design level, they are a bad choice because they say nothing about the values contained within. In general one should prefer a struct for these cases.</fr:p><fr:p>A slice is more complicated. Any modification to a slices contents within a function is reflected in the original, but using append to change the length of the array is not, even if the slice has a capacity greater than its length. This is due to slice being implemented a struct with three fields:</fr:p><fr:ul><fr:li><fr:code>int</fr:code> for length</fr:li>
<fr:li><fr:code>int</fr:code> for capcity</fr:li>
<fr:li>a pointer to a block of memory</fr:li></fr:ul></fr:mainmatter><fr:backmatter /></fr:tree></fr:mainmatter><fr:backmatter /></fr:tree><fr:tree
toc="false"
numbered="false"
show-heading="true"
show-metadata="true"
expanded="false"
root="false"
xmlns:fr="http://www.jonmsterling.com/jms-005P.xml"><fr:frontmatter><fr:anchor>292</fr:anchor><fr:addr
type="user">ocs-000N</fr:addr><fr:route>ocs-000N.xml</fr:route><fr:title
text="Enums in Go">Enums in Go</fr:title><fr:taxon>Note</fr:taxon><fr:date><fr:year>2026</fr:year><fr:month>3</fr:month><fr:day>2</fr:day></fr:date><fr:authors /></fr:frontmatter><fr:mainmatter><fr:p>Go does not have enums like other languages e.g. with an `enum`
keyword, but it's fairly straight forward to implement and extend the
same functionality.</fr:p><fr:p>The following defines the type that will enumerate our variables,
and then we construct the enumeration as constants.</fr:p><fr:pre><![CDATA[type Environment int

const (
      OPS Environment = iota
      UAT
      SITS
)]]></fr:pre><fr:p>That we can create a new type to alias each of these constants,
instead of just doing something like `OPS int = iota`, makes it so
that we can actually reliably use these enumerated values in a type
safe way.</fr:p><fr:p>This way we can define methods that only operate on `Environment`
types, instead of just `int` types. E.g. the following methods take a
`Environment` and return a corresponding endpoint/url.</fr:p><fr:pre><![CDATA[func (e Environment) URS_URL() string {
     switch e {
     case OPS:
          return "https://urs.earthdata.nasa.gov"
     case UAT:
          return "https://uat.urs.earthdata.nasa.gov"
     case SITs:
          return "https://sit.urs.earthdata.nasa.gov"
     default:
          return "https://uat.urs.earthdata.nasa.gov"
}

func (e Environment) CMR_Search_URL() string {
     switch e {
     case OPS:
          return "https://cmr.earthdata.nasa.gov/search/"
     case UAT:
          return "https://cmr.uat.earthdata.nasa.gov/search/"
     case OPS:
          return "https://cmr.sit.earthdata.nasa.gov/search/"
     default:
          return "https://cmr.uat.earthdata.nasa.gov/search/"
}]]></fr:pre><fr:p>Since the underlying type of the enums values are just `int`s, we can also use them just as we would `int`s.</fr:p><fr:pre><![CDATA[func (e Environment) String() string {
     names := [...]string{"OPS", "UAT", "SITS"}
     if e < OPS || e > SITS {
            return fmt.Sprintf("Unkown(%d)", e)
     }
     return names[e]
}]]></fr:pre></fr:mainmatter><fr:backmatter /></fr:tree></fr:mainmatter><fr:backmatter /></fr:tree></fr:mainmatter><fr:backmatter /></fr:tree>