deploy.py auto-commit
1
.gitignore
vendored
|
|
@ -1 +0,0 @@
|
|||
public
|
||||
0
.gitmodules
vendored
|
|
@ -1,6 +0,0 @@
|
|||
---
|
||||
title: "{{ replace .Name "-" " " | title }}"
|
||||
date: {{ .Date }}
|
||||
draft: true
|
||||
---
|
||||
|
||||
|
|
@ -1,4 +0,0 @@
|
|||
baseURL = "https://blog.jaseg.de/"
|
||||
languageCode = "en-us"
|
||||
title = "jaseg.de"
|
||||
theme = "hugo-classic"
|
||||
|
|
@ -1,6 +0,0 @@
|
|||
---
|
||||
title: "About jaseg"
|
||||
---
|
||||
|
||||
About
|
||||
-----
|
||||
|
|
@ -1,102 +0,0 @@
|
|||
---
|
||||
title: "Impressum"
|
||||
---
|
||||
|
||||
Impressum
|
||||
---------
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<p>
|
||||
Sebastian Götte<br/>
|
||||
c/o Praxis Dr. Götte<br/>
|
||||
Krankenhausstr. 1a<br/>
|
||||
54634 Bitburg<br/>
|
||||
imprint@jaseg.net
|
||||
</p>
|
||||
|
||||
Inhaltlich Verantwortlicher gem. § 55 II RStV: Sebastian Götte (Anschrift s.o.)
|
||||
|
||||
Lizenz dieser Seite
|
||||
-------------------
|
||||
|
||||
.. raw:: html
|
||||
|
||||
Dieses Werk ist lizenziert unter einer
|
||||
<a rel="license" href="http://creativecommons.org/licenses/by-nc-sa/4.0/">
|
||||
Creative Commons Namensnennung - Nicht-kommerziell - Weitergabe unter gleichen Bedingungen 4.0 International
|
||||
Lizenz (CC-BY-NC-SA)
|
||||
</a>.<br/>
|
||||
<a rel="license" href="http://creativecommons.org/licenses/by-nc-sa/4.0/">
|
||||
<img alt="Creative Commons Lizenzvertrag" style="border-width:0"
|
||||
src="https://i.creativecommons.org/l/by-nc-sa/4.0/88x31.png" />
|
||||
</a>
|
||||
|
||||
|
||||
Haftungsbeschränkung für Inhalte der Website
|
||||
--------------------------------------------
|
||||
|
||||
Gemäß § 7 Abs. 1 TMG ist der Verantwortliche der Website i. S. v. § 5 TMG für eigene Informationen, die er zur Nutzung
|
||||
bereithält, nach den allgemeinen Gesetzen verantwortlich.
|
||||
|
||||
Der Verantwortliche der vorbezeichneten Website übernimmt keine Haftung auf Aktualität, Richtigkeit und Vollständigkeit
|
||||
der auf dieser Website zur Verfügung gestellten Inhalte. Dies gilt nicht, wenn dem Verantwortlichen vorsätzliches oder
|
||||
grob fahrlässiges Verhalten vorzuwerfen ist. Die Inhalte wurden mit der größtmöglichen Sorgfalt erstellt. Dennoch kann
|
||||
die inhaltliche Richtigkeit insbesondere bei komplexen Themen nicht gewährleistet werden, so dass der Verantwortliche
|
||||
den Nutzern empfiehlt, bei wichtigen Informationen bei den zuständigen Stellen anzufragen oder rechtliche Beratung in
|
||||
Anspruch zu nehmen. Sofern kostenpflichtige Inhalte oder Dienste auf der Website zur Verfügung gestellt werden, handelt
|
||||
es sich dabei um unverbindliche Invitatio ad offerendum, welche lediglich zur Abgabe eines Angebots durch den Nutzer
|
||||
aufrufen und selbst kein verbindliches Angebot darstellen.
|
||||
|
||||
Gemäß §§ 8 ff. TMG ist der Website-Betreiber für fremde Inhalte, die er für einen Nutzer veröffentlicht, nicht
|
||||
verantwortlich, sofern
|
||||
|
||||
* er keine Kenntnis von der rechtswidrigen Handlung oder der Information hat und ihm im Falle von
|
||||
Schadensersatzansprüchen auch keine Tatsachen oder Umstände bekannt sind, aus denen die rechtswidrige Handlung oder
|
||||
die Information offensichtlich wird, oder
|
||||
|
||||
* er unverzüglich tätig geworden ist, um die Information zu entfernen oder den Zugang zu ihr zu sperren, sobald er
|
||||
diese Kenntnis erlangt hat.
|
||||
|
||||
Erlangt der Website-Betreiber Kenntnis von solchen rechtswidrigen Inhalten, werden diese unverzüglich entfernt.
|
||||
|
||||
Haftung für ausgehende Links
|
||||
----------------------------
|
||||
|
||||
Der Website-Betreiber verlinkt von seiner Website auf fremde Websites. Durch diese sog. „Hyperlinks“ wird der Nutzer
|
||||
direkt auf die fremde Website geleitet. Der Website-Betreiber hat dabei keinerlei Einfluss auf die Informationen der
|
||||
fremden Website. Daher kann keine Haftung für die Aktualität, Richtigkeit und Vollständigkeit der Inhalte der fremden
|
||||
Website übernommen werden. Der Website-Betreiber versichert jedoch, dass ihm zum Zeitpunkt des Setzens der Verlinkung
|
||||
keinerlei rechtliche Verstöße bekannt waren und er die fremde Website im Rahmen des Zumutbaren geprüft hat.
|
||||
|
||||
Erhält der Website-Betreiber Kenntnis von der Rechtswidrigkeit der verlinkten Inhalte, wird der entsprechende Link
|
||||
unverzüglich entfernt.
|
||||
|
||||
Urheberrechte
|
||||
-------------
|
||||
|
||||
Die auf dieser Webseite veröffentlichten Inhalte unterliegen dem deutschen Urheberrecht.
|
||||
|
||||
Als Urheber i. S. v. § 7 UrhG stehen dem Website-Betreiber die alleinigen und ausschließlichen Verwertungsrechte gemäß
|
||||
§§ 15 ff. UrhG zu. Eine Vervielfältigung oder Verwendung sämtlicher Inhalte der Website in fremden elektronischen oder
|
||||
gedruckten Medien ist ohne vorherige Zustimmung des Website-Betreibers untersagt und wird ggf. verfolgt.
|
||||
|
||||
Datenschutz
|
||||
-----------
|
||||
|
||||
Die Website kann grundsätzlich ohne Eingabe von personenbezogenen Daten wie z. B. Name oder E-Mail-Adresse genutzt
|
||||
werden. Sollte die Möglichkeit der Eingabe solcher Daten bestehen, so erfolgt die Mitteilung dieser Daten auf
|
||||
freiwilliger Basis durch den Nutzer. Eine Weitergabe dieser Daten an Dritte ist ohne die ausdrückliche Zustimmung durch
|
||||
den Nutzer ausgeschlossen.
|
||||
|
||||
Ein lückenloser Schutz der übermittelten Daten vor dem Zugriff durch unbefugte Dritte ist jedoch nicht möglich, da es im
|
||||
Bereich der Datenübermittlung im Internet zu Sicherheitslücken kommen kann. Der Website-Betreiber versucht jedoch, die
|
||||
Gefahr des unbefugten Zugriffs durch geeignete Maßnahmen zu unterbinden und im Falle der Kenntnis von einer
|
||||
Sicherheitslücke diese durch geeignete Maßnahmen zu schließen.
|
||||
|
||||
Der Nutzung der im Impressum veröffentlichten Kontaktdaten durch Dritte zur Übersendung von nicht ausdrücklich
|
||||
gewünschter Werbung, insbesondere Spam-Mails, wird widersprochen und im Falle der Nichtbeachtung ggf. mit rechtlichen
|
||||
Schritten verfolgt.
|
||||
|
||||
(Quelle: `BUSE HERZ GRUNST Rechtsanwälte <https://www.kanzlei-wirtschaftsrecht.berlin/aktuelles/kostenloser-muster-disclaimer>`__)
|
||||
|
||||
|
|
@ -1,209 +0,0 @@
|
|||
---
|
||||
title: "Hardware Security Module Basics"
|
||||
date: 2019-05-17T15:29:20+08:00
|
||||
---
|
||||
|
||||
Hardware Security Modules and Security Research and Cryptography
|
||||
================================================================
|
||||
|
||||
On May 17 2019 I gave a short presentation on the fundamentals of hardware security modules at the weekly seminar of
|
||||
Prof. Mori's security research working group at Waseda University. The motivation for this was that outside of low-level
|
||||
hardware security people and people working in the financial industry HSMs are not thought about that often. In
|
||||
particular most network or systems security people would not consider them an option. Also it could turn out to be
|
||||
really interesting to think about what could be done with an HSM in conjunction with modern cryptography (instead of
|
||||
just plain old RSA-OAEP and AES-CBC).
|
||||
|
||||
`Click here to download a PDF with the slides for this talk. <mori_semi_hsm_talk_web.pdf>`__
|
||||
|
||||
Ideas for research in HSMs
|
||||
==========================
|
||||
|
||||
Preparing for this talk brought me back to some research ideas I've been working on for a while now. Since I'm not sure
|
||||
I'll find the time to properly research this topic, I thought it would be great to write down some rought outlines first
|
||||
for future reference.
|
||||
|
||||
The Problem with current HSM tech
|
||||
---------------------------------
|
||||
|
||||
Currently, HSMs are only used in certain specific niche applications such as certificate authority key management and
|
||||
financial transaction data handling. One key reason for this is that HSMs currently don't provide the affordances that
|
||||
would be needed for them to be adopted more widely by the cryptographic and security engineering community. As far as I
|
||||
can tell, the two core missing affordances are:
|
||||
|
||||
1. To be more widely adopted, HSMs must become less expensive. Currently, they go for several tens of thousands of Euro,
|
||||
which puts them outside most budgets.
|
||||
2. To be more widely adopted, HSMs must provide the standardized programming interfaces familiar to cryptographic
|
||||
developers. Currently, every HSM vendor has their own custom cryptographic API and a developer will have to train on
|
||||
one specific vendor's tooling. Furthermore, any documentation of these internals is kept secret behind NDAs. This
|
||||
constitutes a high barrier to entry, decreasing adoption in particular with young developers accustomed to
|
||||
open-source ecosystems.
|
||||
|
||||
Attacking cost of implementation
|
||||
--------------------------------
|
||||
|
||||
The first issue can be addressed by simply creating a viable low-cost alternative. There is no fundamental technical
|
||||
reason for the high cost of HSMs. This cost is instead due to manufacturers trying to recoup their expenses for R&D as
|
||||
well as certification from the small volumes HSMs are sold in.
|
||||
|
||||
Compared to system integration and certification the pure R&D cost of HSM defense mechanisms themselves is not too high
|
||||
in an academic context it should be feasible to develop a sort of HSM blueprint that can then be cheaply produced by
|
||||
anyone in need. Since the application areas outlined here are far from the core business areas of the clients of
|
||||
established HSM vendors this would most likely not be a realistic threat to any established vendor's business and a
|
||||
co-existence of both should not pose any problems in the short term.
|
||||
|
||||
Benefits of an academic HSM standard
|
||||
------------------------------------
|
||||
|
||||
Tackling the high cost of current HSM hardware with an open-source HSM blueprint would yield
|
||||
several academic advantages beyond cost reduction.
|
||||
|
||||
1. An open-source blueprint could serve as an academic reference design to evaluate and compare other HSM designs
|
||||
against. For instance this would not only allow quantifying the effectiveness of academic security measures but also
|
||||
allow an evaluation of commercial HSMs.
|
||||
2. An open-source blueprint could stimulate academic research in this academically very quiet albeit commercially
|
||||
important area. This research would ultimately benefit everyone employing HSMs by raising security standards in the
|
||||
field. Since HSMs are never solely relied upon for overal system security both defensive and offensive security
|
||||
research would yield these benefits.
|
||||
3. An open-source blueprint would encourage new people to get into the field and both apply HSMs to practical problems
|
||||
as well as improve HSMs themselves. Currently, this is highly discouraged due to the strictly proprietary nature of
|
||||
all available systems.
|
||||
4. Finally, developing an open-source HSM blueprint might yield new findings in adjacent academic areas due to the
|
||||
hightly multi-disciplinary nature of security research in general and HSM design in particular.
|
||||
|
||||
Scope of an academic HSM standard
|
||||
---------------------------------
|
||||
|
||||
An academic HSM blueprint would need to be flexible so that researchers can adapt it to their particular problem. A
|
||||
modular architecture would lend itself to this flexibility. Fundamentally, there would be three components to this
|
||||
architecture. First, a **base** containing infrastructure such as the surveillance microcontroller, power supplies,
|
||||
power supply filtering and hardware DPA countermeasures, and possibly a standardized mechanical and electrical
|
||||
interface.
|
||||
|
||||
Next to the base, a system integrator would put their *payload*. The nature of this payload is intentionally kept
|
||||
unspecified, and it might be anything from a cryptographic microcontroller to a small embedded system such as a
|
||||
raspberry pi single board computer. Keeping the *payload* open like this achieves two benefits: It gives the HSM
|
||||
blueprint's user *their* familiar tooling and the hardware *they* need, allowing fast adoption. Someone well-versed in
|
||||
e.g. Javascript could literally implement their cryptography in Javascript, run it on an off-the-shelf raspberry pi and
|
||||
just apply the HSM blueprint around it. In addition, keeping the *payload* open reduces the scope of what needs to be
|
||||
implemented. Building a general SDK on top of something like a bare ARM SoC such as a TI OMAP or a Freescale/NXP IMX
|
||||
would be a considerable additional burden, on top of the actual HSM design. Keeping the *payload* open allows research
|
||||
to concentrate on the actual point, the HSM design.
|
||||
|
||||
The final and most important component would be a set of *security measures* that can be combined with the base to
|
||||
form the final HSM. Each of these *security measures* would entail a detailed specification of its design, manufacture
|
||||
and security properties. These *security measures* could be simple things like tamper switches or potting, but could
|
||||
also be complex things like security meshes.
|
||||
|
||||
Given these three components -- *base*, *payload* and *security measures* as detailed specifications any engineer should
|
||||
be able to design and manufacture a HSM customized to their needs. Unifying these three components within the HSM
|
||||
blueprint would be a set of reference designs. Each reference design would implement a particular parametrization of the
|
||||
three architectural components with a physical hole cut out where the payload would go.. These reference designs would
|
||||
for one serve to guide any implementer on the customization and integration of their own derivation from the blueprint.
|
||||
In addition it would serve as an extremely simple, low-cost point of entry into the ecosystem. A curious researcher
|
||||
could simply replicate the reference design and put their existing payload inside. Practically this might mean e.g. a
|
||||
researcher having PCBs produced according to the design files for a reference design for a mesh-based HSM, producing
|
||||
their own mesh, physically glueing a raspberry pi SBC into the middle of it, and potting the resulting system. Given the
|
||||
ease of prototype PCB fabrication today this would realistically allow evaluation of HSM technologies on a budget that
|
||||
is orders of magnitude less than the cost of current HSMs.
|
||||
|
||||
Research ideas for tamper detection mechanisms
|
||||
==============================================
|
||||
|
||||
The core component of an HSM blueprint would be a suite of tamper detection mechanisms. Following are a few ideas on how
|
||||
to improve on the current state of the art of membrane tamper switches plus temperature sensors plus PCB and printed
|
||||
security meshes plus potting.
|
||||
|
||||
Improvements on existing techniques
|
||||
-----------------------------------
|
||||
|
||||
Light sensors
|
||||
~~~~~~~~~~~~~
|
||||
**Advanced analog sensing**
|
||||
**Self-test functionality**
|
||||
|
||||
Security meshes
|
||||
~~~~~~~~~~~~~~~
|
||||
**Analog sensing**
|
||||
|
||||
|
||||
DIY or small lab mesh production
|
||||
--------------------------------
|
||||
**3D metal patterning techniques** refers to any technique for producing thin, patterned metal structures on a
|
||||
three-dimensional plastic substrate. The basic process would consist of 3D-printing the polymer substrate, depositing a
|
||||
thin metal layer on top and then patterning this metal layer. A good starting point here would be the recent work of
|
||||
`Ben Kraznow`_ on this exact thing.
|
||||
|
||||
**Copper filament methods** would be any method embedding copper wire from a spool in some resin or other matrix. This
|
||||
could mean either of a systematic approach of carefully winding or folding the copper wire into patterns or a
|
||||
non-systematic approach of simply stuffing a large tangle of copper wire into a small space. The main challenge with the
|
||||
former would be to find a non-tedious way of production. The main challenge with the latter would be to find process
|
||||
parameters that guarantee complete coverage of the HSM without holes or other areas of lower sensitivity to intrusions.
|
||||
Both approaches would require careful consideration of the overall design including the polymer resin supporting
|
||||
structure to ensure sensitivity against attacks since copper wire is mechanically much stronger than the micrometre-thin
|
||||
metal coatings used in patterning techniques.
|
||||
|
||||
Envelope measurement
|
||||
--------------------
|
||||
|
||||
Finally, I think there is another set of currently under-utilized tamper-detection methods that would be very
|
||||
interesting to explore. I am not aware of an academic term for these, so I am just going to dub them *envelope
|
||||
measurement* here.
|
||||
|
||||
The fundamental apporach of a mesh is to build a physical security envelope (the mesh) that physically detects when it
|
||||
is disturbed (open or short circuits). This approach works well but has the disadvantage that these meshes are rather
|
||||
complex to manufacture since effectively every part of them is acting as a sensing element. A conceptually more complex
|
||||
but in practice potentially simpler approach might be to split the functions of security envelope and sensing element.
|
||||
This would mean that in place of the mesh, some form of passive element such as metal foil forms the security envelope
|
||||
which is then checked for tampering using a very sensitive sensor inside. This remote-sensing approach might simplify
|
||||
the manufacture of the envelope itself and thus yield a design that is more easily customized. Following are a few ideas
|
||||
on how to approach this envelope measurement problem.
|
||||
|
||||
**Ultrasonic** If the HSM is potted, a few ultrasonic transducers could be added inside the potting. With several
|
||||
transducers, any one could be used to transmit ultrasound while the others measure complex phase and energy of the
|
||||
signal they receive. The circuitry for this could be made fairly simple if using a static transmit frequency or a low
|
||||
chirp rate by using a homodyne receiver built around a comparator fed into some timers. This approach would likely
|
||||
detect any mechanical attack and would also rule out chemical attacks involving liquids (though starting from which
|
||||
amount of liquid depends on receiver sensitivity). The main disadvantages might be high power consumption and cost and
|
||||
size of the ultrasonic transducers. Traditional cheap transducers made for air as a transmission medium are fairly large
|
||||
and might not adequately couple into potting compound. If somehow one could convince a standard small piezo element to
|
||||
do the same job that would be great as far as cost and size are concerned. A concern in some fringe use cases might be
|
||||
suceptibility to ambient noise, though this could easily be reduced at the expense of space and heat dissipation
|
||||
capacity by adding sound dampening on the outside. A likely attack vector against this approach might be using a laser
|
||||
cutter to drill a hole through the potting compound, then inserting probes carefully chosen to not couple too much
|
||||
to the potting compound ultrasonically.
|
||||
|
||||
**Light** In either an unpotted HSM or one potted with a transparent (at some wavelengths) potting compound one could
|
||||
embed LEDs and photodiodes in a similar setup to the ultrasonic setup described above. In contrast to the ultrasound,
|
||||
the LEDs would literally have to light up the HSM's interior and shadows might be an issue since the HSM is likely some
|
||||
flat rectangular shape. A possible solution to this would be to coat both the embedded payload and the lid with some
|
||||
highly reflective paint such as some glossy silver paint or simple white paint. The basic approach might be as simple as
|
||||
simply turning on several LEDs distributed throughout the HSM in turn and measuring amplitude at several photodetectors,
|
||||
or as complex as doing a LIDAR-like phase measurement sweeping through a range of frequencies to determine not only
|
||||
absorption but also phase/distance characteristics between emitter LED and detector photodiode. Using some high-gain TIA
|
||||
along with a homodyne detector (lock-in amplifier) and changing emitter intensity, very precise measurements of both
|
||||
absorption and phase might be possible, as might be measurements through almost opaque, diffuse potting compounds such
|
||||
as a grey epoxide resin. The main disadvantages of this method would likely be the need to thoroughly light-proof the
|
||||
entire HSM (likely by wrapping it in metal foil) and the potentially high cost of transmitter and receiver circuitry
|
||||
(nice TIAs aren't cheap). To be effective against attacks using e.g. very fine drills and probes the system would likely
|
||||
have to be very sensitive.
|
||||
|
||||
**Radar** Finally, one could turn to standard radar techniques to fingerprint the inside of the HSM. The goal here would
|
||||
be fingerprinting instead of mapping since only changes need to be detected. In this approach one could use homodyne
|
||||
detection to improve sensitivity and reduce receiver complexity, and sweep frequencies similar to an FMCW radar (but
|
||||
probably without exploiting the self-demodulation effect). Besides high cost, this approach has two disadvantages.
|
||||
First, such a system would likely not go beyond 24GHz or maybe 40GHz due to component availability issues. Even at 40GHz
|
||||
the wavelength in the potting compound would be in the order of magnitude of several millimeters. Fine intrusions using
|
||||
some tool chosen to not interact too much with the EM field inside the HSM such as a heated ceramic needle or simply a
|
||||
laser cutter might not be detectable using this approach. In any case, this system would certainly not be able to detect
|
||||
small holes piercing the HSM enclosure. The HSM enclosure would have to be made into an RF shield, likely by using some
|
||||
metal foil in it.
|
||||
|
||||
Overall in the author's opinion these three techniques are most promising in order *Light*, *Ultrasonic*, *Radar*. Light
|
||||
would prbably provide the best sensitivity at expense of some cost. Ultrasonic might be used in conjunction with light
|
||||
to cover some additional angles since it is potentially very low-cost. Radar seems hard to engineer into a solution that
|
||||
works reliably and also would likely be at least an order of magnitude more expensive than the other two technologies
|
||||
while not providing better sensitivity.
|
||||
|
||||
.. _`Ben Kraznow`: https://www.youtube.com/watch?v=Z228xymQYho
|
||||
.. _affordances: https://en.wikipedia.org/wiki/Affordance
|
||||
|
||||
|
Before Width: | Height: | Size: 35 KiB |
|
Before Width: | Height: | Size: 292 KiB |
|
Before Width: | Height: | Size: 296 KiB |
|
Before Width: | Height: | Size: 296 KiB |
|
Before Width: | Height: | Size: 296 KiB |
|
Before Width: | Height: | Size: 296 KiB |
|
Before Width: | Height: | Size: 46 KiB |
|
Before Width: | Height: | Size: 46 KiB |
|
Before Width: | Height: | Size: 118 KiB |
|
Before Width: | Height: | Size: 193 KiB |
|
Before Width: | Height: | Size: 45 KiB |
|
Before Width: | Height: | Size: 128 KiB |
|
Before Width: | Height: | Size: 189 KiB |
|
Before Width: | Height: | Size: 175 KiB |
|
Before Width: | Height: | Size: 104 KiB |
|
Before Width: | Height: | Size: 1.7 MiB |
|
Before Width: | Height: | Size: 809 KiB |
|
|
@ -1,221 +0,0 @@
|
|||
---
|
||||
title: "Kicad Mesh Plugin"
|
||||
date: 2020-08-18T13:15:39+02:00
|
||||
---
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/anim.webp" style="max-width: 20em">
|
||||
</figure>
|
||||
|
||||
Tamper Detection Meshes
|
||||
=======================
|
||||
|
||||
Cryptography is at the foundation of our modern, networked world. From email to card payment infrastructure in brick and
|
||||
mortar stores, cryptographic keys secure almost every part of our digital lives againts cybercriminals or curious
|
||||
surveillance capitalists. Without cryptography, many of the things we routinely do in our lives such as paying for
|
||||
groceries with a credit card, messaging a friend on `Signal <https://signal.org>`_ or unlocking a car with its keyfob
|
||||
would not be possible. The security of all of these systems in its core lies on the secrecy of cryptographic keys.
|
||||
Systems differ in what kind of keys they use, how often these keys are replaced and the intricacies of the cryptographic
|
||||
operations these keys fit into but all have in common that their security relies on keeping the keys secret.
|
||||
|
||||
In practice, this secrecy has been implemented in many different ways. Mass-market software such as browsers or
|
||||
messenger apps usually relies on some operating system facility to tell the computer "*please keep this piece of memory
|
||||
away from all other applications*". While on desktop operating systems usually this does not provide much of a barrier
|
||||
to other programs on the same computer, on modern mobile operating systems this approach is actually quite secure.
|
||||
However, given sufficient resources no security is perfect. All of these systems can be compromised if the host
|
||||
operating system is compromised sufficiently, and for organizations with considerable resources a market has sprung up
|
||||
that offers turn-key solutions for all wiretapping needs.
|
||||
|
||||
In some applications, this level of security has not been considered sufficient. Particularly financial infrastructure
|
||||
is such a high-profile target that a lot of effort has been put into the security of cryptographic implementations. The
|
||||
best cryptographic algorithm is useless if it is run on a compromised system (from that system's point of view anyway).
|
||||
One of the core cryptographic components in financial applications are smartcards like they are used as payment cards in
|
||||
most countries nowadays. These smartcards contain a small, specialized cryptographic microcontroller that is designed to
|
||||
be hard to tamper with. Though one of the design goals of the system is to reduce the amount of sensitive information
|
||||
stored on the card, things such as copying of a card can only be hindered by making the chip hard to read out.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/modern_art.svg" style="max-width: 20em">
|
||||
</figure>
|
||||
|
||||
With smartcards being the means of choice on one side of the counter in electronic payments, on the other side of the
|
||||
counter a different technology prevails. Attacks on payment terminals are bound to have much more dire consequences than
|
||||
attacks on individual cards since one terminal might see hundreds of cards being read every day. For this reason, the
|
||||
level of attack countermeasures employed in these terminals is a considerable step up from bare smartcards. While a
|
||||
smartcard is made physically hard to tamper, it does not have a battery and it can only detect tampering once it is
|
||||
powered by a reader. This allows for well-equipped attackers to use tools such as Focused Ion Beam (FIB) workstations to
|
||||
circumvent the smartcard's defences while it is powered down, and then power up the card to carry out the actual attack.
|
||||
|
||||
The answer to this problem in electronic payment infrastructure is called *Hardware Security Module*, or HSM. An HSM is
|
||||
similar to a smartcard in its function (cryptographic processing using keys that are meant to never leave the protection
|
||||
of the HSM). The one major between the two is that an HSM has its own battery and is continuously powered from its
|
||||
manufacture to the day it is scrapped. If the HSM looses power at any point in time, it uses a small amount of energy
|
||||
stored internally to securely wipe all cryptographic secrets from its memory within a few milliseconds.
|
||||
|
||||
Being powered at all times allows the HSM to actively detect and respond to attacks. The most common way this is done is
|
||||
by wrapping the juicy secret parts in a foil or a printed circuit board that is patterned with a long and convoluted
|
||||
maze of wires, called a *mesh*. The HSM is continuously monitoring these wires for changes (such as shorts, breaks or
|
||||
changes in resistance) and will sound the alarm when any are detected. Practically, this presents a considerable hurdle
|
||||
to any attacker: They have to find a way to disable or circumvent the mesh while it is being monitored by the HSM. In
|
||||
practice, often this is no insurmountable challenge but it again increases attack costs.
|
||||
|
||||
DIY Meshes
|
||||
==========
|
||||
|
||||
Throughout my studies in security research I have always had an interest in HSMs. I have taken apart my fair share of
|
||||
HSMs and at this point, to understand the technology more, I want to experiment with building my own HSM. In last year's
|
||||
`HSM basics <{{<ref "posts/hsm-basics/index.rst">}}>`_ post I have lined out some ideas for a next generation design that
|
||||
deviates from the bread-and-butter apporoach of using a mesh as the primary security feature. Before embarking on
|
||||
practical experiments with these ideas, I want to first take a stab at replicating the current state of the art as best
|
||||
I can. State of the art meshes often use exotic substrates such as 3D plastic parts with traces chemically deposited on
|
||||
their surface or special flexible substrates with conductive ink traces. These technologies will likely be too
|
||||
cumbersome for me to implement myself only for a few prototypes, and industrial manufacturers will most likely be too
|
||||
expensive. Thus, I will concentrate on regular PCB technology for now.
|
||||
|
||||
The idea of a mesh on a PCB is pretty simple: You have one or several traces that you try to cover every corner of the
|
||||
mesh PCB's area with. To be most effective, the traces should be as thin and as close together as possible. To increase
|
||||
the chances of a manipulation being detected, multiple traces can also be used that can then be monitored for shorts
|
||||
between them.
|
||||
|
||||
While one can feasibly lay out these traces by hand, this really is an ideal application of a simple auto-router. While
|
||||
general PCB autorouting is *hard*, auto-routing just a few traces to approximate a space-filling curve is not. Since I
|
||||
am just starting out, I went with the simplest algorithmic solution I could think of. I first approximate the area
|
||||
designated to the mesh with a square grid whose cells are a multiple of my trace/space size. The mesh will only be drawn
|
||||
into grid cells that are fully inside the set boundaries. All cells outside or going across the border are discarded in
|
||||
this step.
|
||||
|
||||
I decided to implement this auto-router in a KiCAD plugin. Though KiCADs plugin API is not the best, it was just about
|
||||
usable for this task.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/kicad-mesh-outline.png" alt="KiCAD showing an irregular board shape with rounded corners and
|
||||
indents. In the middle of the board there is a footprint for a 4-pin surface-mount pin header.">
|
||||
<figcaption>The process starts out with the mesh shape being defined inside KiCAD. The mesh's outline is drawn
|
||||
onto one of the graphical "Eco" layers. A footprint is placed to serve as a placeholder for the mesh's
|
||||
connections to the outside world. This footprint is later used as the starting point for the mesh generation
|
||||
algorithm.</figcaption>
|
||||
</figure>
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/grid-vis-plain.svg" alt="A vizualization of the grid fitting process. Over the mesh's irregular
|
||||
outline a grid is drawn. In this picture, all grid cells that are fully inside the grid are shown. Grid cells
|
||||
that overlap the mesh border are highlighted. Grid cells outside of the mesh border are not drawn.">
|
||||
<figcaption>A visualization of the grid fitting process. First, a grid large enough to contain the mesh border
|
||||
is generated. Then, every cell is checked for overlap with the mesh border area. If the cell is fully inside, it
|
||||
(yellow), it is considered in the mesh generation later. Cells outside (gray) or on the border (red) are
|
||||
discarded.</figcaption>
|
||||
</figure>
|
||||
|
||||
After generating the grid, starting from the place I want to connect to the mesh, I walk the grid's cells one by one to
|
||||
generate a tree that covers the entire grid's area. To set the mesh's starting place I place a footprint on the board
|
||||
(dark gray in the picture above) whose designator I then tell my script. The tree generation algorithm looks like a
|
||||
depth-first search, except all checks are random. Depending on the level of randomness used at each step of the
|
||||
algorithm it yields more or less organized-looking results. Below are five example runs of the algorithm at differing
|
||||
levels of randomness with the cells colored according to their distance from the tree root. 0% randomness means that the
|
||||
algorithm is going to try cells in forward direction first on every step, and only then try out left and right. 100%
|
||||
means that on every step, the algorithm is choosing a new direction at random.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<figure class="side-by-side">
|
||||
<img src="images/cells-0.svg" alt="a completely organized looking grid with spiral patterns all over.">
|
||||
<figcaption>0%</figcaption>
|
||||
</figure><figure class="side-by-side">
|
||||
<img src="images/cells-25.svg">
|
||||
<figcaption>25%</figcaption>
|
||||
</figure><figure class="side-by-side">
|
||||
<img src="images/cells-50.svg">
|
||||
<figcaption>50%</figcaption>
|
||||
</figure><figure class="side-by-side">
|
||||
<img src="images/cells-75.svg">
|
||||
<figcaption>75%</figcaption>
|
||||
</figure><figure class="side-by-side">
|
||||
<img src="images/cells-100.svg" alt="a completely random looking grid with cells aggregating into ireggular
|
||||
areas that look like paint splotches.">
|
||||
<figcaption>100%</figcaption>
|
||||
</figure>
|
||||
</figure>
|
||||
|
||||
After I have built this tree like you would do in a depth-first search, I draw my one or several mesh mesh traces into
|
||||
it. The core observation here is that there is only 16 possible ways a cell can be connected: It has four neighbors,
|
||||
each of which it can either be connected to or not, which results in 2^4 options. If you consider rotations and
|
||||
mirroring, this works out to rotations or mirrored versions of only six base tiles: The empty tile, a tile with all four
|
||||
sides connected, a straight through, a 90 degree bend, and a "T"-junction—see the illustration below.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/maze_tiles_plain.svg" style="max-width: 20em">
|
||||
<figcaption>
|
||||
There are six possible tile types in our connectivity graph inside its square tiling. This graphic illustrates
|
||||
all sixteen rotations of these with how they would look in a two-conductor mesh.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
After tiling the grid according to the key above, we get the result below.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/tiles-25-small.svg">
|
||||
<figcaption>
|
||||
An auto-routed mesh with traces colored according to tile types.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/traces-25-small.svg">
|
||||
<figcaption>
|
||||
The same mesh, but with traces all black.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
Putting it all together got me the KiCAD plugin you can see in the screenshot below.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/kicad-mesh-settings2.png">
|
||||
<figcaption>
|
||||
The plugin settings window open.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/kicad-mesh-result-large.png">
|
||||
<figcaption>
|
||||
After runing the plugin, the generated mesh looks like this in pcbnew.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
I am fairly happy with the result, but getting there was a medium pain. Especially KiCAD's plugin API is still very
|
||||
unfinieshed. It is hard to use, most parts are completely undocumented and if you use anything but its most basic parts
|
||||
things tend to break. One particular pain point for me was that after generating the mesh, the traces have been added to
|
||||
the board, but are still invisible for some reason. You have to save the board first, then re-load the file for them to
|
||||
become visible. Also KiCAD crashes whenever the plugin tries to remove a trace, so currently my workflow involves always
|
||||
making a copy of the board file first and treating mesh generation as a non-reversible finishing step.
|
||||
|
||||
`Check out the code on my cgit <https://git.jaseg.de/kimesh.git/tree/plugin/mesh_dialog.py>`_.
|
||||
|
||||
.. ::
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/grid-vis-plain.svg" alt="">
|
||||
<figcaption></figcaption>
|
||||
</figure>
|
||||
|
||||
|
Before Width: | Height: | Size: 77 KiB |
|
Before Width: | Height: | Size: 279 KiB |
|
Before Width: | Height: | Size: 285 KiB |
|
Before Width: | Height: | Size: 428 KiB |
|
Before Width: | Height: | Size: 287 KiB |
|
|
@ -1,874 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8" standalone="no"?>
|
||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN"
|
||||
"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||
<!-- Created with matplotlib (http://matplotlib.org/) -->
|
||||
<svg height="345pt" version="1.1" viewBox="0 0 460 345" width="460pt" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||
<defs>
|
||||
<style type="text/css">
|
||||
*{stroke-linecap:butt;stroke-linejoin:round;}
|
||||
</style>
|
||||
</defs>
|
||||
<g id="figure_1">
|
||||
<g id="patch_1">
|
||||
<path d="M 0 345.6
|
||||
L 460.8 345.6
|
||||
L 460.8 0
|
||||
L 0 0
|
||||
z
|
||||
" style="fill:none;opacity:0;"/>
|
||||
</g>
|
||||
<g id="axes_1">
|
||||
<g id="patch_2">
|
||||
<path d="M 57.6 307.584
|
||||
L 414.72 307.584
|
||||
L 414.72 41.472
|
||||
L 57.6 41.472
|
||||
z
|
||||
" style="fill:#ffffff;"/>
|
||||
</g>
|
||||
<g id="matplotlib.axis_1">
|
||||
<g id="xtick_1">
|
||||
<g id="line2d_1">
|
||||
<path clip-path="url(#pade65162c6)" d="M 78.607059 307.584
|
||||
L 78.607059 41.472
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_2">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L 0 3.5
|
||||
" id="mc8506604b7" style="stroke:#01769d;stroke-width:0.8;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="78.607059" xlink:href="#mc8506604b7" y="307.584"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_1">
|
||||
<!-- 400 -->
|
||||
<defs>
|
||||
<path d="M 37.796875 64.3125
|
||||
L 12.890625 25.390625
|
||||
L 37.796875 25.390625
|
||||
z
|
||||
M 35.203125 72.90625
|
||||
L 47.609375 72.90625
|
||||
L 47.609375 25.390625
|
||||
L 58.015625 25.390625
|
||||
L 58.015625 17.1875
|
||||
L 47.609375 17.1875
|
||||
L 47.609375 0
|
||||
L 37.796875 0
|
||||
L 37.796875 17.1875
|
||||
L 4.890625 17.1875
|
||||
L 4.890625 26.703125
|
||||
z
|
||||
" id="DejaVuSans-34"/>
|
||||
<path d="M 31.78125 66.40625
|
||||
Q 24.171875 66.40625 20.328125 58.90625
|
||||
Q 16.5 51.421875 16.5 36.375
|
||||
Q 16.5 21.390625 20.328125 13.890625
|
||||
Q 24.171875 6.390625 31.78125 6.390625
|
||||
Q 39.453125 6.390625 43.28125 13.890625
|
||||
Q 47.125 21.390625 47.125 36.375
|
||||
Q 47.125 51.421875 43.28125 58.90625
|
||||
Q 39.453125 66.40625 31.78125 66.40625
|
||||
M 31.78125 74.21875
|
||||
Q 44.046875 74.21875 50.515625 64.515625
|
||||
Q 56.984375 54.828125 56.984375 36.375
|
||||
Q 56.984375 17.96875 50.515625 8.265625
|
||||
Q 44.046875 -1.421875 31.78125 -1.421875
|
||||
Q 19.53125 -1.421875 13.0625 8.265625
|
||||
Q 6.59375 17.96875 6.59375 36.375
|
||||
Q 6.59375 54.828125 13.0625 64.515625
|
||||
Q 19.53125 74.21875 31.78125 74.21875
|
||||
" id="DejaVuSans-30"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(69.063309 322.182437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-34"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-30"/>
|
||||
<use x="127.246094" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_2">
|
||||
<g id="line2d_3">
|
||||
<path clip-path="url(#pade65162c6)" d="M 131.124706 307.584
|
||||
L 131.124706 41.472
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_4">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="131.124706" xlink:href="#mc8506604b7" y="307.584"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_2">
|
||||
<!-- 450 -->
|
||||
<defs>
|
||||
<path d="M 10.796875 72.90625
|
||||
L 49.515625 72.90625
|
||||
L 49.515625 64.59375
|
||||
L 19.828125 64.59375
|
||||
L 19.828125 46.734375
|
||||
Q 21.96875 47.46875 24.109375 47.828125
|
||||
Q 26.265625 48.1875 28.421875 48.1875
|
||||
Q 40.625 48.1875 47.75 41.5
|
||||
Q 54.890625 34.8125 54.890625 23.390625
|
||||
Q 54.890625 11.625 47.5625 5.09375
|
||||
Q 40.234375 -1.421875 26.90625 -1.421875
|
||||
Q 22.3125 -1.421875 17.546875 -0.640625
|
||||
Q 12.796875 0.140625 7.71875 1.703125
|
||||
L 7.71875 11.625
|
||||
Q 12.109375 9.234375 16.796875 8.0625
|
||||
Q 21.484375 6.890625 26.703125 6.890625
|
||||
Q 35.15625 6.890625 40.078125 11.328125
|
||||
Q 45.015625 15.765625 45.015625 23.390625
|
||||
Q 45.015625 31 40.078125 35.4375
|
||||
Q 35.15625 39.890625 26.703125 39.890625
|
||||
Q 22.75 39.890625 18.8125 39.015625
|
||||
Q 14.890625 38.140625 10.796875 36.28125
|
||||
z
|
||||
" id="DejaVuSans-35"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(121.580956 322.182437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-34"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-35"/>
|
||||
<use x="127.246094" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_3">
|
||||
<g id="line2d_5">
|
||||
<path clip-path="url(#pade65162c6)" d="M 183.642353 307.584
|
||||
L 183.642353 41.472
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_6">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="183.642353" xlink:href="#mc8506604b7" y="307.584"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_3">
|
||||
<!-- 500 -->
|
||||
<g style="fill:#01769d;" transform="translate(174.098603 322.182437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-35"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-30"/>
|
||||
<use x="127.246094" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_4">
|
||||
<g id="line2d_7">
|
||||
<path clip-path="url(#pade65162c6)" d="M 236.16 307.584
|
||||
L 236.16 41.472
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_8">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="236.16" xlink:href="#mc8506604b7" y="307.584"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_4">
|
||||
<!-- 550 -->
|
||||
<g style="fill:#01769d;" transform="translate(226.61625 322.182437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-35"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-35"/>
|
||||
<use x="127.246094" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_5">
|
||||
<g id="line2d_9">
|
||||
<path clip-path="url(#pade65162c6)" d="M 288.677647 307.584
|
||||
L 288.677647 41.472
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_10">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="288.677647" xlink:href="#mc8506604b7" y="307.584"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_5">
|
||||
<!-- 600 -->
|
||||
<defs>
|
||||
<path d="M 33.015625 40.375
|
||||
Q 26.375 40.375 22.484375 35.828125
|
||||
Q 18.609375 31.296875 18.609375 23.390625
|
||||
Q 18.609375 15.53125 22.484375 10.953125
|
||||
Q 26.375 6.390625 33.015625 6.390625
|
||||
Q 39.65625 6.390625 43.53125 10.953125
|
||||
Q 47.40625 15.53125 47.40625 23.390625
|
||||
Q 47.40625 31.296875 43.53125 35.828125
|
||||
Q 39.65625 40.375 33.015625 40.375
|
||||
M 52.59375 71.296875
|
||||
L 52.59375 62.3125
|
||||
Q 48.875 64.0625 45.09375 64.984375
|
||||
Q 41.3125 65.921875 37.59375 65.921875
|
||||
Q 27.828125 65.921875 22.671875 59.328125
|
||||
Q 17.53125 52.734375 16.796875 39.40625
|
||||
Q 19.671875 43.65625 24.015625 45.921875
|
||||
Q 28.375 48.1875 33.59375 48.1875
|
||||
Q 44.578125 48.1875 50.953125 41.515625
|
||||
Q 57.328125 34.859375 57.328125 23.390625
|
||||
Q 57.328125 12.15625 50.6875 5.359375
|
||||
Q 44.046875 -1.421875 33.015625 -1.421875
|
||||
Q 20.359375 -1.421875 13.671875 8.265625
|
||||
Q 6.984375 17.96875 6.984375 36.375
|
||||
Q 6.984375 53.65625 15.1875 63.9375
|
||||
Q 23.390625 74.21875 37.203125 74.21875
|
||||
Q 40.921875 74.21875 44.703125 73.484375
|
||||
Q 48.484375 72.75 52.59375 71.296875
|
||||
" id="DejaVuSans-36"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(279.133897 322.182437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-36"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-30"/>
|
||||
<use x="127.246094" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_6">
|
||||
<g id="line2d_11">
|
||||
<path clip-path="url(#pade65162c6)" d="M 341.195294 307.584
|
||||
L 341.195294 41.472
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_12">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="341.195294" xlink:href="#mc8506604b7" y="307.584"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_6">
|
||||
<!-- 650 -->
|
||||
<g style="fill:#01769d;" transform="translate(331.651544 322.182437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-36"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-35"/>
|
||||
<use x="127.246094" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_7">
|
||||
<g id="line2d_13">
|
||||
<path clip-path="url(#pade65162c6)" d="M 393.712941 307.584
|
||||
L 393.712941 41.472
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_14">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="393.712941" xlink:href="#mc8506604b7" y="307.584"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_7">
|
||||
<!-- 700 -->
|
||||
<defs>
|
||||
<path d="M 8.203125 72.90625
|
||||
L 55.078125 72.90625
|
||||
L 55.078125 68.703125
|
||||
L 28.609375 0
|
||||
L 18.3125 0
|
||||
L 43.21875 64.59375
|
||||
L 8.203125 64.59375
|
||||
z
|
||||
" id="DejaVuSans-37"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(384.169191 322.182437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-37"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-30"/>
|
||||
<use x="127.246094" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_8">
|
||||
<!-- $\lambda\;[nm]$ -->
|
||||
<defs>
|
||||
<path d="M 36.71875 67.4375
|
||||
L 48.828125 0
|
||||
L 39.3125 0
|
||||
L 31.84375 40.4375
|
||||
L 5.125 0
|
||||
L -4.390625 0
|
||||
L 29.734375 52.4375
|
||||
L 28.03125 62.109375
|
||||
Q 26.953125 68.265625 21.734375 68.265625
|
||||
L 17.046875 68.265625
|
||||
L 18.5 75.984375
|
||||
L 24.21875 75.875
|
||||
Q 35.203125 75.734375 36.71875 67.4375
|
||||
" id="DejaVuSans-Oblique-3bb"/>
|
||||
<path d="M 8.59375 75.984375
|
||||
L 29.296875 75.984375
|
||||
L 29.296875 69
|
||||
L 17.578125 69
|
||||
L 17.578125 -6.203125
|
||||
L 29.296875 -6.203125
|
||||
L 29.296875 -13.1875
|
||||
L 8.59375 -13.1875
|
||||
z
|
||||
" id="DejaVuSans-5b"/>
|
||||
<path d="M 55.71875 33.015625
|
||||
L 49.3125 0
|
||||
L 40.28125 0
|
||||
L 46.6875 32.671875
|
||||
Q 47.125 34.96875 47.359375 36.71875
|
||||
Q 47.609375 38.484375 47.609375 39.5
|
||||
Q 47.609375 43.609375 45.015625 45.890625
|
||||
Q 42.4375 48.1875 37.796875 48.1875
|
||||
Q 30.5625 48.1875 25.34375 43.375
|
||||
Q 20.125 38.578125 18.5 30.328125
|
||||
L 12.5 0
|
||||
L 3.515625 0
|
||||
L 14.109375 54.6875
|
||||
L 23.09375 54.6875
|
||||
L 21.296875 46.09375
|
||||
Q 25.046875 50.828125 30.3125 53.40625
|
||||
Q 35.59375 56 41.40625 56
|
||||
Q 48.640625 56 52.609375 52.09375
|
||||
Q 56.59375 48.1875 56.59375 41.109375
|
||||
Q 56.59375 39.359375 56.375 37.359375
|
||||
Q 56.15625 35.359375 55.71875 33.015625
|
||||
" id="DejaVuSans-Oblique-6e"/>
|
||||
<path d="M 89.796875 33.015625
|
||||
L 83.40625 0
|
||||
L 74.421875 0
|
||||
L 80.71875 32.71875
|
||||
Q 81.109375 34.8125 81.296875 36.328125
|
||||
Q 81.5 37.84375 81.5 38.921875
|
||||
Q 81.5 43.3125 79.046875 45.75
|
||||
Q 76.609375 48.1875 72.21875 48.1875
|
||||
Q 65.671875 48.1875 60.546875 43.28125
|
||||
Q 55.421875 38.375 53.90625 30.515625
|
||||
L 47.90625 0
|
||||
L 38.921875 0
|
||||
L 45.3125 32.71875
|
||||
Q 45.703125 34.515625 45.890625 36.046875
|
||||
Q 46.09375 37.59375 46.09375 38.8125
|
||||
Q 46.09375 43.265625 43.65625 45.71875
|
||||
Q 41.21875 48.1875 36.921875 48.1875
|
||||
Q 30.28125 48.1875 25.140625 43.28125
|
||||
Q 20.015625 38.375 18.5 30.515625
|
||||
L 12.5 0
|
||||
L 3.515625 0
|
||||
L 14.203125 54.6875
|
||||
L 23.1875 54.6875
|
||||
L 21.484375 46.1875
|
||||
Q 25.140625 50.984375 30.046875 53.484375
|
||||
Q 34.96875 56 40.578125 56
|
||||
Q 46.53125 56 50.359375 52.875
|
||||
Q 54.203125 49.75 54.984375 44.1875
|
||||
Q 59.078125 49.953125 64.46875 52.96875
|
||||
Q 69.875 56 75.875 56
|
||||
Q 82.90625 56 86.734375 51.953125
|
||||
Q 90.578125 47.90625 90.578125 40.484375
|
||||
Q 90.578125 38.875 90.375 36.9375
|
||||
Q 90.1875 35.015625 89.796875 33.015625
|
||||
" id="DejaVuSans-Oblique-6d"/>
|
||||
<path d="M 30.421875 75.984375
|
||||
L 30.421875 -13.1875
|
||||
L 9.71875 -13.1875
|
||||
L 9.71875 -6.203125
|
||||
L 21.390625 -6.203125
|
||||
L 21.390625 69
|
||||
L 9.71875 69
|
||||
L 9.71875 75.984375
|
||||
z
|
||||
" id="DejaVuSans-5d"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(219.86 335.860562)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.015625)" xlink:href="#DejaVuSans-Oblique-3bb"/>
|
||||
<use transform="translate(86.238823 0.015625)" xlink:href="#DejaVuSans-5b"/>
|
||||
<use transform="translate(125.252495 0.015625)" xlink:href="#DejaVuSans-Oblique-6e"/>
|
||||
<use transform="translate(188.631401 0.015625)" xlink:href="#DejaVuSans-Oblique-6d"/>
|
||||
<use transform="translate(286.043511 0.015625)" xlink:href="#DejaVuSans-5d"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="matplotlib.axis_2">
|
||||
<g id="ytick_1">
|
||||
<g id="line2d_15">
|
||||
<path clip-path="url(#pade65162c6)" d="M 57.6 295.226473
|
||||
L 414.72 295.226473
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_16">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L -3.5 0
|
||||
" id="m3646c9359b" style="stroke:#01769d;stroke-width:0.8;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="57.6" xlink:href="#m3646c9359b" y="295.226473"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_9">
|
||||
<!-- 0.2 -->
|
||||
<defs>
|
||||
<path d="M 10.6875 12.40625
|
||||
L 21 12.40625
|
||||
L 21 0
|
||||
L 10.6875 0
|
||||
z
|
||||
" id="DejaVuSans-2e"/>
|
||||
<path d="M 19.1875 8.296875
|
||||
L 53.609375 8.296875
|
||||
L 53.609375 0
|
||||
L 7.328125 0
|
||||
L 7.328125 8.296875
|
||||
Q 12.9375 14.109375 22.625 23.890625
|
||||
Q 32.328125 33.6875 34.8125 36.53125
|
||||
Q 39.546875 41.84375 41.421875 45.53125
|
||||
Q 43.3125 49.21875 43.3125 52.78125
|
||||
Q 43.3125 58.59375 39.234375 62.25
|
||||
Q 35.15625 65.921875 28.609375 65.921875
|
||||
Q 23.96875 65.921875 18.8125 64.3125
|
||||
Q 13.671875 62.703125 7.8125 59.421875
|
||||
L 7.8125 69.390625
|
||||
Q 13.765625 71.78125 18.9375 73
|
||||
Q 24.125 74.21875 28.421875 74.21875
|
||||
Q 39.75 74.21875 46.484375 68.546875
|
||||
Q 53.21875 62.890625 53.21875 53.421875
|
||||
Q 53.21875 48.921875 51.53125 44.890625
|
||||
Q 49.859375 40.875 45.40625 35.40625
|
||||
Q 44.1875 33.984375 37.640625 27.21875
|
||||
Q 31.109375 20.453125 19.1875 8.296875
|
||||
" id="DejaVuSans-32"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(34.696875 299.025692)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-30"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-2e"/>
|
||||
<use x="95.410156" xlink:href="#DejaVuSans-32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_2">
|
||||
<g id="line2d_17">
|
||||
<path clip-path="url(#pade65162c6)" d="M 57.6 264.114344
|
||||
L 414.72 264.114344
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_18">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="57.6" xlink:href="#m3646c9359b" y="264.114344"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_10">
|
||||
<!-- 0.3 -->
|
||||
<defs>
|
||||
<path d="M 40.578125 39.3125
|
||||
Q 47.65625 37.796875 51.625 33
|
||||
Q 55.609375 28.21875 55.609375 21.1875
|
||||
Q 55.609375 10.40625 48.1875 4.484375
|
||||
Q 40.765625 -1.421875 27.09375 -1.421875
|
||||
Q 22.515625 -1.421875 17.65625 -0.515625
|
||||
Q 12.796875 0.390625 7.625 2.203125
|
||||
L 7.625 11.71875
|
||||
Q 11.71875 9.328125 16.59375 8.109375
|
||||
Q 21.484375 6.890625 26.8125 6.890625
|
||||
Q 36.078125 6.890625 40.9375 10.546875
|
||||
Q 45.796875 14.203125 45.796875 21.1875
|
||||
Q 45.796875 27.640625 41.28125 31.265625
|
||||
Q 36.765625 34.90625 28.71875 34.90625
|
||||
L 20.21875 34.90625
|
||||
L 20.21875 43.015625
|
||||
L 29.109375 43.015625
|
||||
Q 36.375 43.015625 40.234375 45.921875
|
||||
Q 44.09375 48.828125 44.09375 54.296875
|
||||
Q 44.09375 59.90625 40.109375 62.90625
|
||||
Q 36.140625 65.921875 28.71875 65.921875
|
||||
Q 24.65625 65.921875 20.015625 65.03125
|
||||
Q 15.375 64.15625 9.8125 62.3125
|
||||
L 9.8125 71.09375
|
||||
Q 15.4375 72.65625 20.34375 73.4375
|
||||
Q 25.25 74.21875 29.59375 74.21875
|
||||
Q 40.828125 74.21875 47.359375 69.109375
|
||||
Q 53.90625 64.015625 53.90625 55.328125
|
||||
Q 53.90625 49.265625 50.4375 45.09375
|
||||
Q 46.96875 40.921875 40.578125 39.3125
|
||||
" id="DejaVuSans-33"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(34.696875 267.913563)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-30"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-2e"/>
|
||||
<use x="95.410156" xlink:href="#DejaVuSans-33"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_3">
|
||||
<g id="line2d_19">
|
||||
<path clip-path="url(#pade65162c6)" d="M 57.6 233.002216
|
||||
L 414.72 233.002216
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_20">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="57.6" xlink:href="#m3646c9359b" y="233.002216"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_11">
|
||||
<!-- 0.4 -->
|
||||
<g style="fill:#01769d;" transform="translate(34.696875 236.801435)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-30"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-2e"/>
|
||||
<use x="95.410156" xlink:href="#DejaVuSans-34"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_4">
|
||||
<g id="line2d_21">
|
||||
<path clip-path="url(#pade65162c6)" d="M 57.6 201.890087
|
||||
L 414.72 201.890087
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_22">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="57.6" xlink:href="#m3646c9359b" y="201.890087"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_12">
|
||||
<!-- 0.5 -->
|
||||
<g style="fill:#01769d;" transform="translate(34.696875 205.689306)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-30"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-2e"/>
|
||||
<use x="95.410156" xlink:href="#DejaVuSans-35"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_5">
|
||||
<g id="line2d_23">
|
||||
<path clip-path="url(#pade65162c6)" d="M 57.6 170.777958
|
||||
L 414.72 170.777958
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_24">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="57.6" xlink:href="#m3646c9359b" y="170.777958"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_13">
|
||||
<!-- 0.6 -->
|
||||
<g style="fill:#01769d;" transform="translate(34.696875 174.577177)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-30"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-2e"/>
|
||||
<use x="95.410156" xlink:href="#DejaVuSans-36"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_6">
|
||||
<g id="line2d_25">
|
||||
<path clip-path="url(#pade65162c6)" d="M 57.6 139.66583
|
||||
L 414.72 139.66583
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_26">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="57.6" xlink:href="#m3646c9359b" y="139.66583"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_14">
|
||||
<!-- 0.7 -->
|
||||
<g style="fill:#01769d;" transform="translate(34.696875 143.465048)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-30"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-2e"/>
|
||||
<use x="95.410156" xlink:href="#DejaVuSans-37"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_7">
|
||||
<g id="line2d_27">
|
||||
<path clip-path="url(#pade65162c6)" d="M 57.6 108.553701
|
||||
L 414.72 108.553701
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_28">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="57.6" xlink:href="#m3646c9359b" y="108.553701"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_15">
|
||||
<!-- 0.8 -->
|
||||
<defs>
|
||||
<path d="M 31.78125 34.625
|
||||
Q 24.75 34.625 20.71875 30.859375
|
||||
Q 16.703125 27.09375 16.703125 20.515625
|
||||
Q 16.703125 13.921875 20.71875 10.15625
|
||||
Q 24.75 6.390625 31.78125 6.390625
|
||||
Q 38.8125 6.390625 42.859375 10.171875
|
||||
Q 46.921875 13.96875 46.921875 20.515625
|
||||
Q 46.921875 27.09375 42.890625 30.859375
|
||||
Q 38.875 34.625 31.78125 34.625
|
||||
M 21.921875 38.8125
|
||||
Q 15.578125 40.375 12.03125 44.71875
|
||||
Q 8.5 49.078125 8.5 55.328125
|
||||
Q 8.5 64.0625 14.71875 69.140625
|
||||
Q 20.953125 74.21875 31.78125 74.21875
|
||||
Q 42.671875 74.21875 48.875 69.140625
|
||||
Q 55.078125 64.0625 55.078125 55.328125
|
||||
Q 55.078125 49.078125 51.53125 44.71875
|
||||
Q 48 40.375 41.703125 38.8125
|
||||
Q 48.828125 37.15625 52.796875 32.3125
|
||||
Q 56.78125 27.484375 56.78125 20.515625
|
||||
Q 56.78125 9.90625 50.3125 4.234375
|
||||
Q 43.84375 -1.421875 31.78125 -1.421875
|
||||
Q 19.734375 -1.421875 13.25 4.234375
|
||||
Q 6.78125 9.90625 6.78125 20.515625
|
||||
Q 6.78125 27.484375 10.78125 32.3125
|
||||
Q 14.796875 37.15625 21.921875 38.8125
|
||||
M 18.3125 54.390625
|
||||
Q 18.3125 48.734375 21.84375 45.5625
|
||||
Q 25.390625 42.390625 31.78125 42.390625
|
||||
Q 38.140625 42.390625 41.71875 45.5625
|
||||
Q 45.3125 48.734375 45.3125 54.390625
|
||||
Q 45.3125 60.0625 41.71875 63.234375
|
||||
Q 38.140625 66.40625 31.78125 66.40625
|
||||
Q 25.390625 66.40625 21.84375 63.234375
|
||||
Q 18.3125 60.0625 18.3125 54.390625
|
||||
" id="DejaVuSans-38"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(34.696875 112.35292)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-30"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-2e"/>
|
||||
<use x="95.410156" xlink:href="#DejaVuSans-38"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_8">
|
||||
<g id="line2d_29">
|
||||
<path clip-path="url(#pade65162c6)" d="M 57.6 77.441572
|
||||
L 414.72 77.441572
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_30">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="57.6" xlink:href="#m3646c9359b" y="77.441572"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_16">
|
||||
<!-- 0.9 -->
|
||||
<defs>
|
||||
<path d="M 10.984375 1.515625
|
||||
L 10.984375 10.5
|
||||
Q 14.703125 8.734375 18.5 7.8125
|
||||
Q 22.3125 6.890625 25.984375 6.890625
|
||||
Q 35.75 6.890625 40.890625 13.453125
|
||||
Q 46.046875 20.015625 46.78125 33.40625
|
||||
Q 43.953125 29.203125 39.59375 26.953125
|
||||
Q 35.25 24.703125 29.984375 24.703125
|
||||
Q 19.046875 24.703125 12.671875 31.3125
|
||||
Q 6.296875 37.9375 6.296875 49.421875
|
||||
Q 6.296875 60.640625 12.9375 67.421875
|
||||
Q 19.578125 74.21875 30.609375 74.21875
|
||||
Q 43.265625 74.21875 49.921875 64.515625
|
||||
Q 56.59375 54.828125 56.59375 36.375
|
||||
Q 56.59375 19.140625 48.40625 8.859375
|
||||
Q 40.234375 -1.421875 26.421875 -1.421875
|
||||
Q 22.703125 -1.421875 18.890625 -0.6875
|
||||
Q 15.09375 0.046875 10.984375 1.515625
|
||||
M 30.609375 32.421875
|
||||
Q 37.25 32.421875 41.125 36.953125
|
||||
Q 45.015625 41.5 45.015625 49.421875
|
||||
Q 45.015625 57.28125 41.125 61.84375
|
||||
Q 37.25 66.40625 30.609375 66.40625
|
||||
Q 23.96875 66.40625 20.09375 61.84375
|
||||
Q 16.21875 57.28125 16.21875 49.421875
|
||||
Q 16.21875 41.5 20.09375 36.953125
|
||||
Q 23.96875 32.421875 30.609375 32.421875
|
||||
" id="DejaVuSans-39"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(34.696875 81.240791)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-30"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-2e"/>
|
||||
<use x="95.410156" xlink:href="#DejaVuSans-39"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_9">
|
||||
<g id="line2d_31">
|
||||
<path clip-path="url(#pade65162c6)" d="M 57.6 46.329444
|
||||
L 414.72 46.329444
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_32">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="57.6" xlink:href="#m3646c9359b" y="46.329444"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_17">
|
||||
<!-- 1.0 -->
|
||||
<defs>
|
||||
<path d="M 12.40625 8.296875
|
||||
L 28.515625 8.296875
|
||||
L 28.515625 63.921875
|
||||
L 10.984375 60.40625
|
||||
L 10.984375 69.390625
|
||||
L 28.421875 72.90625
|
||||
L 38.28125 72.90625
|
||||
L 38.28125 8.296875
|
||||
L 54.390625 8.296875
|
||||
L 54.390625 0
|
||||
L 12.40625 0
|
||||
z
|
||||
" id="DejaVuSans-31"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(34.696875 50.128662)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-31"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-2e"/>
|
||||
<use x="95.410156" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_18">
|
||||
<!-- $S_{rel,820nm}\;[1]$ -->
|
||||
<defs>
|
||||
<path d="M 60.296875 70.515625
|
||||
L 58.40625 60.890625
|
||||
Q 53.46875 63.53125 48.609375 64.875
|
||||
Q 43.75 66.21875 39.203125 66.21875
|
||||
Q 30.375 66.21875 25.140625 62.359375
|
||||
Q 19.921875 58.5 19.921875 52.09375
|
||||
Q 19.921875 48.578125 21.84375 46.703125
|
||||
Q 23.78125 44.828125 31.78125 42.671875
|
||||
L 37.703125 41.21875
|
||||
Q 47.703125 38.625 51.609375 34.640625
|
||||
Q 55.515625 30.671875 55.515625 23.484375
|
||||
Q 55.515625 12.453125 46.84375 5.515625
|
||||
Q 38.1875 -1.421875 24.03125 -1.421875
|
||||
Q 18.21875 -1.421875 12.359375 -0.265625
|
||||
Q 6.5 0.875 0.59375 3.21875
|
||||
L 2.59375 13.375
|
||||
Q 8.015625 10.015625 13.453125 8.296875
|
||||
Q 18.890625 6.59375 24.3125 6.59375
|
||||
Q 33.546875 6.59375 39.109375 10.6875
|
||||
Q 44.671875 14.796875 44.671875 21.390625
|
||||
Q 44.671875 25.78125 42.453125 28.046875
|
||||
Q 40.234375 30.328125 32.90625 32.171875
|
||||
L 27 33.6875
|
||||
Q 16.890625 36.328125 13.203125 39.765625
|
||||
Q 9.515625 43.21875 9.515625 49.421875
|
||||
Q 9.515625 60.296875 17.890625 67.25
|
||||
Q 26.265625 74.21875 39.703125 74.21875
|
||||
Q 44.921875 74.21875 50.046875 73.28125
|
||||
Q 55.171875 72.359375 60.296875 70.515625
|
||||
" id="DejaVuSans-Oblique-53"/>
|
||||
<path d="M 44.578125 46.390625
|
||||
Q 43.21875 47.125 41.453125 47.515625
|
||||
Q 39.703125 47.90625 37.703125 47.90625
|
||||
Q 30.515625 47.90625 25.140625 42.453125
|
||||
Q 19.78125 37.015625 18.015625 27.875
|
||||
L 12.5 0
|
||||
L 3.515625 0
|
||||
L 14.203125 54.6875
|
||||
L 23.1875 54.6875
|
||||
L 21.484375 46.1875
|
||||
Q 25.046875 50.921875 30 53.453125
|
||||
Q 34.96875 56 40.578125 56
|
||||
Q 42.046875 56 43.453125 55.828125
|
||||
Q 44.875 55.671875 46.296875 55.28125
|
||||
z
|
||||
" id="DejaVuSans-Oblique-72"/>
|
||||
<path d="M 48.09375 32.234375
|
||||
Q 48.25 33.015625 48.3125 33.84375
|
||||
Q 48.390625 34.671875 48.390625 35.5
|
||||
Q 48.390625 41.453125 44.890625 44.921875
|
||||
Q 41.40625 48.390625 35.40625 48.390625
|
||||
Q 28.71875 48.390625 23.578125 44.15625
|
||||
Q 18.453125 39.9375 15.828125 32.171875
|
||||
z
|
||||
M 55.90625 25.203125
|
||||
L 14.109375 25.203125
|
||||
Q 13.8125 23.34375 13.71875 22.265625
|
||||
Q 13.625 21.1875 13.625 20.40625
|
||||
Q 13.625 13.625 17.796875 9.90625
|
||||
Q 21.96875 6.203125 29.59375 6.203125
|
||||
Q 35.453125 6.203125 40.671875 7.515625
|
||||
Q 45.90625 8.84375 50.390625 11.375
|
||||
L 48.6875 2.484375
|
||||
Q 43.84375 0.53125 38.6875 -0.4375
|
||||
Q 33.546875 -1.421875 28.21875 -1.421875
|
||||
Q 16.84375 -1.421875 10.71875 4.015625
|
||||
Q 4.59375 9.46875 4.59375 19.484375
|
||||
Q 4.59375 28.03125 7.640625 35.375
|
||||
Q 10.6875 42.71875 16.609375 48.484375
|
||||
Q 20.40625 52.09375 25.65625 54.046875
|
||||
Q 30.90625 56 36.8125 56
|
||||
Q 46.09375 56 51.578125 50.4375
|
||||
Q 57.078125 44.875 57.078125 35.5
|
||||
Q 57.078125 33.25 56.78125 30.6875
|
||||
Q 56.5 28.125 55.90625 25.203125
|
||||
" id="DejaVuSans-Oblique-65"/>
|
||||
<path d="M 18.3125 75.984375
|
||||
L 27.296875 75.984375
|
||||
L 12.5 0
|
||||
L 3.515625 0
|
||||
z
|
||||
" id="DejaVuSans-Oblique-6c"/>
|
||||
<path d="M 11.71875 12.40625
|
||||
L 22.015625 12.40625
|
||||
L 22.015625 4
|
||||
L 14.015625 -11.625
|
||||
L 7.71875 -11.625
|
||||
L 11.71875 4
|
||||
z
|
||||
" id="DejaVuSans-2c"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(28.196875 204.978)rotate(-90)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.015625)" xlink:href="#DejaVuSans-Oblique-53"/>
|
||||
<use transform="translate(63.476562 -16.390625)scale(0.7)" xlink:href="#DejaVuSans-Oblique-72"/>
|
||||
<use transform="translate(92.255859 -16.390625)scale(0.7)" xlink:href="#DejaVuSans-Oblique-65"/>
|
||||
<use transform="translate(135.322266 -16.390625)scale(0.7)" xlink:href="#DejaVuSans-Oblique-6c"/>
|
||||
<use transform="translate(154.770508 -16.390625)scale(0.7)" xlink:href="#DejaVuSans-2c"/>
|
||||
<use transform="translate(190.65918 -16.390625)scale(0.7)" xlink:href="#DejaVuSans-38"/>
|
||||
<use transform="translate(235.195312 -16.390625)scale(0.7)" xlink:href="#DejaVuSans-32"/>
|
||||
<use transform="translate(279.731445 -16.390625)scale(0.7)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(324.267578 -16.390625)scale(0.7)" xlink:href="#DejaVuSans-Oblique-6e"/>
|
||||
<use transform="translate(368.632812 -16.390625)scale(0.7)" xlink:href="#DejaVuSans-Oblique-6d"/>
|
||||
<use transform="translate(466.6148 0.015625)" xlink:href="#DejaVuSans-5b"/>
|
||||
<use transform="translate(505.628472 0.015625)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(569.251519 0.015625)" xlink:href="#DejaVuSans-5d"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="line2d_33">
|
||||
<path clip-path="url(#pade65162c6)" d="M 57.6 295.488
|
||||
L 108.016941 258.421055
|
||||
L 130.074353 241.875537
|
||||
L 152.131765 225.042643
|
||||
L 176.289882 206.308347
|
||||
L 215.152941 175.819697
|
||||
L 242.462118 154.525609
|
||||
L 260.318118 140.872387
|
||||
L 275.023059 129.893794
|
||||
L 287.627294 120.736002
|
||||
L 299.181176 112.591191
|
||||
L 310.735059 104.726261
|
||||
L 321.238588 97.853297
|
||||
L 330.691765 91.919526
|
||||
L 340.144941 86.247859
|
||||
L 348.547765 81.444735
|
||||
L 356.950588 76.882481
|
||||
L 365.353412 72.576869
|
||||
L 373.756235 68.543515
|
||||
L 381.108706 65.249786
|
||||
L 388.461176 62.186461
|
||||
L 395.813647 59.363515
|
||||
L 403.166118 56.790707
|
||||
L 410.518588 54.477559
|
||||
L 413.669647 53.568
|
||||
L 413.669647 53.568
|
||||
" style="fill:none;stroke:#fe3ea0;stroke-linecap:square;stroke-width:1.5;"/>
|
||||
</g>
|
||||
<g id="patch_3">
|
||||
<path d="M 57.6 307.584
|
||||
L 57.6 41.472
|
||||
" style="fill:none;stroke:#08bdf9;stroke-linecap:square;stroke-linejoin:miter;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="patch_4">
|
||||
<path d="M 57.6 307.584
|
||||
L 414.72 307.584
|
||||
" style="fill:none;stroke:#08bdf9;stroke-linecap:square;stroke-linejoin:miter;stroke-width:0.8;"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<defs>
|
||||
<clipPath id="pade65162c6">
|
||||
<rect height="266.112" width="357.12" x="57.6" y="41.472"/>
|
||||
</clipPath>
|
||||
</defs>
|
||||
</svg>
|
||||
|
Before Width: | Height: | Size: 29 KiB |
|
Before Width: | Height: | Size: 332 KiB |
|
Before Width: | Height: | Size: 301 KiB |
|
Before Width: | Height: | Size: 271 KiB |
|
Before Width: | Height: | Size: 88 KiB |
|
Before Width: | Height: | Size: 103 KiB |
|
|
@ -1,222 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!-- Generator: Adobe Illustrator 13.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 14948) -->
|
||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||
<svg version="1.1" id="Ebene_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" width="1092px" height="306px" viewBox="5 5 728 204" enable-background="new 5 5 728 204">
|
||||
|
||||
<defs>
|
||||
|
||||
<marker id="white_arrow_head" markerWidth="6" markerHeight="6" refX="10" refY="6" orient="auto" viewBox="0 0 12 12">
|
||||
<path d="M0,0 v12 L12,6 z" fill="white"/>
|
||||
</marker>
|
||||
|
||||
<marker id="black_arrow_head" markerWidth="6" markerHeight="6" refX="10" refY="6" orient="auto" viewBox="0 0 12 12">
|
||||
<path d="M0,0 v12 L12,6 z" fill="black"/>
|
||||
</marker>
|
||||
|
||||
<style type="text/css">
|
||||
|
||||
.inner_grid_on_dark {stroke:white}
|
||||
.inner_grid_on_bright {stroke:black}
|
||||
.outer_grid {stroke:black; fill:none}
|
||||
|
||||
text {font-family:DejaVu Sans,Liberation Sans, Arial, sans-serif}
|
||||
text.inner_axis_on_dark {font-size:32px; text-anchor:middle; fill:white}
|
||||
text.inner_axis_on_bright {font-size:32px; text-anchor:middle; fill:black}
|
||||
.explanation_text { font-size:9px; text-anchor:end; fill:white}
|
||||
text.outer_axis {font-size:22px; text-anchor:middle}
|
||||
text.outer_axis_smaller {font-size:14px; text-anchor:middle}
|
||||
|
||||
.explanation_line {stroke:white;}
|
||||
.explanation_arrow {stroke:white;fill:none; marker-end: url(#white_arrow_head)}
|
||||
.inner_arrow_on_dark {stroke-width:3px;fill:none; stroke:white; marker-end: url(#white_arrow_head)}
|
||||
.inner_arrow_on_bright {stroke-width:3px;fill:none; stroke:black; marker-end: url(#black_arrow_head)}
|
||||
|
||||
.mini_grid { stroke:white; stroke-dasharray:1; fill:none }
|
||||
.explanation_circle { fill:#539642; stroke:white; stroke-width:2 }
|
||||
|
||||
</style>
|
||||
|
||||
<symbol id="B_GB" viewBox="-128 -128 256 256">
|
||||
<image overflow="visible" width="256" height="256" xlink:href="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAQAAAAEACAYAAABccqhmAAAABHNCSVQICAgIfAhkiAAAAAlwSFlz AAAPYQAAD2EBqD+naQAAAx5JREFUeJzt1AEJAEEQxLBZeP+W74U0gVrobXvbTVKvbzcgygAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAgzAAg zAAgzAAgzAAgzAAgzAAg7AeEIAb7+p29LAAAAABJRU5ErkJggg==" transform="matrix(1 0 0 1 -128 -128)"/>
|
||||
</symbol>
|
||||
|
||||
<symbol id="B_RB" viewBox="-128 -128 256 256">
|
||||
<image overflow="visible" width="256" height="256" xlink:href="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAQAAAAEACAYAAABccqhmAAAABHNCSVQICAgIfAhkiAAAAAlwSFlz AAAPYQAAD2EBqD+naQAAAx1JREFUeJzt1AEJAEEQxLBZeP+W74U0gVrobXvbTVKv7wZUGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCE/YUfBvtEaIT/AAAAAElFTkSuQmCC" transform="matrix(1 0 0 1 -128 -128)"/>
|
||||
</symbol>
|
||||
|
||||
<symbol id="B_RG" viewBox="-128 -128 256 256">
|
||||
<image overflow="visible" width="256" height="256" xlink:href="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAQAAAAEACAYAAABccqhmAAAABHNCSVQICAgIfAhkiAAAAAlwSFlz AAAPYQAAD2EBqD+naQAAAx1JREFUeJzt1AEJAEEQxLBZeP+W74U0gVrobXu7TVKv7wZUGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCEGQCE GQCEGQCEGQCEGQCEGQCE/YYeBvtUDHeuAAAAAElFTkSuQmCC" transform="matrix(1 0 0 1 -128 -128)"/>
|
||||
</symbol>
|
||||
|
||||
<symbol id="W_B" viewBox="-128 -128 256 256">
|
||||
|
||||
<image overflow="visible" width="256" height="256" xlink:href="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAQAAAAEACAYAAABccqhmAAAABHNCSVQICAgIfAhkiAAAAAlwSFlz AAAPYQAAD2EBqD+naQAAAx5JREFUeJzt1AEJAEEQxLB9eP+W54Q0gVrot223O0m9/h1QZQAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQ9gCo8wQa1SrwUwAAAABJRU5ErkJggg==" transform="matrix(1 0 0 1 -128 -128)"/>
|
||||
</symbol>
|
||||
|
||||
<symbol id="W_G" viewBox="-128 -128 256 256">
|
||||
<image overflow="visible" width="256" height="256" xlink:href="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAQAAAAEACAYAAABccqhmAAAABHNCSVQICAgIfAhkiAAAAAlwSFlz AAAPYQAAD2EBqD+naQAAAx5JREFUeJzt1AEJAEEQxLB9eP+W54Q0gVrot213O0m9/h1QZQAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQZgAQ ZgAQZgAQZgAQZgAQZgAQ9gCq4wQafrStoQAAAABJRU5ErkJggg==" transform="matrix(1 0 0 1 -128 -128)">
|
||||
</image>
|
||||
</symbol>
|
||||
|
||||
<symbol id="W_R" viewBox="-128 -128 256 256">
|
||||
<image overflow="visible" width="256" height="256" xlink:href="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAQAAAAEACAYAAABccqhmAAAABHNCSVQICAgIfAhkiAAAAAlwSFlz AAAPYQAAD2EBqD+naQAAAx5JREFUeJzt1AEJAEEQxLB9eP+W54Q0gVrot213O0m9/tsBUQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYAYQYA YQYAYQYAYQYAYQYAYQYAYQ+s0wQaf94YmgAAAABJRU5ErkJggg==" transform="matrix(1 0 0 1 -128 -128)">
|
||||
</image>
|
||||
</symbol>
|
||||
|
||||
</defs>
|
||||
|
||||
<g>
|
||||
|
||||
<g id="cube_1" class="black_cube">
|
||||
|
||||
<g class="transformed_faces">
|
||||
|
||||
<g transform="matrix(-0.2885 0.1055 0 -0.483 -286.9285 -297.8369) matrix(1 0 0 1 -1226.6912 -1101.9382)">
|
||||
<use xlink:href="#B_RG" width="256" height="256" x="-128" y="-128" overflow="visible"/>
|
||||
<g transform="rotate(180) translate(0,110)">
|
||||
<text class="inner_axis_on_dark">G+</text>
|
||||
<path class="inner_arrow_on_dark" d="M30,5 h-75"/>
|
||||
</g>
|
||||
</g>
|
||||
|
||||
<g transform="matrix(0.408 7.449998e-002 0 -0.483 -197.7751 -301.8052) matrix(1 0 0 1 867.4027 -700.2046)">
|
||||
<use xlink:href="#B_RB" width="256" height="256" x="-128" y="-128" overflow="visible"/>
|
||||
<g transform="scale(0.95,-0.95) translate(-90,40)">
|
||||
<text class="inner_axis_on_dark">R+</text>
|
||||
<path class="inner_arrow_on_dark" d="M-30,10 v-55"/>
|
||||
</g>
|
||||
<g transform="scale(0.95,-0.95) translate(-10,118)">
|
||||
<text class="inner_axis_on_dark">B+</text>
|
||||
<path class="inner_arrow_on_dark" d="M-35,5 h70"/>
|
||||
</g>
|
||||
</g>
|
||||
|
||||
<g transform="matrix(0.408 7.449998e-002 -0.2885 0.1055 -234.7039 -226.4771) matrix(1 0 0 1 2379.251 2138.0725)">
|
||||
<use xlink:href="#B_GB" width="256" height="256" x="-128" y="-128" overflow="visible"/>
|
||||
</g>
|
||||
|
||||
</g>
|
||||
|
||||
<g class="grid">
|
||||
<line class="inner_grid_on_dark" x1="208.3" y1="172.4" x2="103.9" y2="153.3"/>
|
||||
<line class="inner_grid_on_dark" x1="103.9" y1="153.3" x2="103.9" y2="29.6"/>
|
||||
<line class="inner_grid_on_dark" x1="103.9" y1="153.3" x2="30" y2="180.3"/>
|
||||
<line class="inner_grid_on_dark" x1="30" y1="56.6" x2="134.5" y2="75.7"/>
|
||||
<line class="inner_grid_on_dark" x1="134.5" y1="75.7" x2="134.5" y2="199.4"/>
|
||||
<line class="inner_grid_on_dark" x1="134.5" y1="75.7" x2="208.3" y2="48.7"/>
|
||||
<polygon class="outer_grid" points="103.9,29.6 208.3,48.7 208.3,172.4 134.5,199.4 30,180.3 30,56.6"/>
|
||||
</g>
|
||||
|
||||
</g>
|
||||
|
||||
|
||||
<g id="cube_2" class="white_cube">
|
||||
|
||||
<g class="transformed_faces">
|
||||
|
||||
<g transform="matrix(-0.408 -7.449998e-002 0.2885 -0.1055 357.554 52.6631)">
|
||||
<use xlink:href="#W_R" width="256" height="256" x="-128" y="-128" overflow="visible"/>
|
||||
</g>
|
||||
|
||||
<g transform="matrix(-0.408 -7.449998e-002 0 0.483 320.6262 127.9912)">
|
||||
<use xlink:href="#W_G" width="256" height="256" x="-128" y="-128" overflow="visible"/>
|
||||
<g transform="scale(-0.95,0.95) translate(90,0)">
|
||||
<text class="inner_axis_on_bright">R-</text>
|
||||
<path class="inner_arrow_on_bright" d="M30,-35 v55"/>
|
||||
</g>
|
||||
<g transform="scale(-0.95,0.95) translate(0,-90)">
|
||||
<text class="inner_axis_on_bright">B-</text>
|
||||
<path class="inner_arrow_on_bright" d="M30,-30 h-70"/>
|
||||
</g>
|
||||
</g>
|
||||
|
||||
<g transform="matrix(0.2885 -0.1055 0 0.483 409.7776 124.0234)">
|
||||
<use xlink:href="#W_B" width="256" height="256" x="-128" y="-128" overflow="visible"/>
|
||||
<g transform="translate(0,-85)">
|
||||
<text class="inner_axis_on_bright">G-</text>
|
||||
<path class="inner_arrow_on_bright" d="M-40,-30 h70"/>
|
||||
</g>
|
||||
</g>
|
||||
|
||||
</g>
|
||||
|
||||
<g class="grid">
|
||||
<line class="inner_grid_on_bright" x1="268.3" y1="56.6" x2="372.8" y2="75.7"/>
|
||||
<line class="inner_grid_on_bright" x1="372.8" y1="75.7" x2="372.8" y2="199.4"/>
|
||||
<line class="inner_grid_on_bright" x1="372.8" y1="75.7" x2="446.6" y2="48.7"/>
|
||||
<polygon class="outer_grid" points="342.3,29.6 446.6,48.7 446.6,172.4 372.8,199.4 268.3,180.3 268.3,56.6"/>
|
||||
</g>
|
||||
|
||||
</g>
|
||||
|
||||
|
||||
<g id="cube_3" class="black_cube">
|
||||
|
||||
<g class="transformed_faces">
|
||||
<use xlink:href="#B_RG" width="256" height="256" x="-128" y="-128" transform="matrix(-0.2885 0.1055 0 -0.483 543.5715 104.9834)" overflow="visible"/>
|
||||
<use xlink:href="#B_RB" width="256" height="256" x="-128" y="-128" transform="matrix(0.408 7.449998e-002 0 -0.483 632.7249 101.0151)" overflow="visible"/>
|
||||
<use xlink:href="#B_GB" width="256" height="256" x="-128" y="-128" transform="matrix(0.408 7.449998e-002 -0.2885 0.1055 595.7961 176.3438)" overflow="visible"/>
|
||||
</g>
|
||||
|
||||
<g class="grid">
|
||||
<line class="inner_grid_on_dark" x1="684.9" y1="172.4" x2="580.5" y2="153.3"/>
|
||||
<line class="inner_grid_on_dark" x1="684.9" y1="172.4" x2="580.5" y2="153.3"/>
|
||||
<line class="inner_grid_on_dark" x1="580.5" y1="153.3" x2="580.5" y2="29.6"/>
|
||||
<line class="inner_grid_on_dark" x1="580.5" y1="153.3" x2="506.6" y2="180.3"/>
|
||||
<line class="inner_grid_on_dark" x1="506.6" y1="56.6" x2="611.1" y2="75.7"/>
|
||||
<line class="inner_grid_on_dark" x1="611.1" y1="75.7" x2="611.1" y2="199.4"/>
|
||||
<line class="inner_grid_on_dark" x1="611.1" y1="75.7" x2="684.9" y2="48.7"/>
|
||||
<polygon class="outer_grid on_dark" points="580.5,29.6 684.9,48.7 684.9,172.4 611.1,199.4 506.6,180.3 506.6,56.6"/>
|
||||
</g>
|
||||
|
||||
<g class="mini_grid">
|
||||
<line x1="607.8" y1="118.3" x2="564.4" y2="134.2"/>
|
||||
<line x1="564.4" y1="134.2" x2="537.1" y2="129.2"/>
|
||||
<line x1="564.4" y1="134.2" x2="564.4" y2="174.2"/>
|
||||
<polygon points="580.5,113.3 607.8,118.3 607.8,158.3 564.4,174.2 537.1,169.2 537.1,129.2"/>
|
||||
</g>
|
||||
|
||||
<g class="explanation">
|
||||
<line class="explanation_line" x1="560.6" y1="129.3" x2="548.8" y2="117"/>
|
||||
<polyline class="explanation_arrow" points="553.4,181.8 544.4,181.8 537.2,171.6"/>
|
||||
<polyline class="explanation_arrow" points="644,151.6 620.1,151.6 610.1,157.2"/>
|
||||
<polyline class="explanation_arrow" points="609,106.5 588.6,106.5 581.7,112.1"/>
|
||||
<circle class="explanation_circle" cx="564.4" cy="134.2" r="7.5"/>
|
||||
<g class="explanation_text">
|
||||
<g transform="matrix(1 0 0 1 535 98)">
|
||||
<text>R</text> <text x="5">:</text> <text x="25">83</text>
|
||||
</g>
|
||||
<g transform="matrix(1 0 0 1 535 106)">
|
||||
<text>G</text> <text x="5">:</text> <text x="25">150</text>
|
||||
</g>
|
||||
<g transform="matrix(1 0 0 1 535 114)">
|
||||
<text>B</text> <text x="5">:</text> <text x="25">60</text>
|
||||
</g>
|
||||
<text transform="matrix(1 0 0 1 585 184)">G: 150</text>
|
||||
<text transform="matrix(1 0 0 1 645 150.208)">B: 60</text>
|
||||
<text transform="matrix(1 0 0 1 610 104.5)">R: 83</text>
|
||||
</g>
|
||||
</g>
|
||||
|
||||
</g>
|
||||
|
||||
<g>
|
||||
<g>
|
||||
<text class="outer_axis" transform="matrix(1 0 0 1 104 26)">R</text>
|
||||
<text class="outer_axis" transform="matrix(1 0 0 1 18 190)">G</text>
|
||||
<text class="outer_axis" transform="matrix(1 0 0 1 219 180)">B</text>
|
||||
</g>
|
||||
<g transform="matrix(1 0 0 1 238 0)">
|
||||
<text class="outer_axis" transform="matrix(1 0 0 1 104 26)">R</text>
|
||||
<text class="outer_axis" transform="matrix(1 0 0 1 18 190)">G</text>
|
||||
<text class="outer_axis" transform="matrix(1 0 0 1 219 180)">B</text>
|
||||
</g>
|
||||
<g transform="matrix(1 0 0 1 475 0)">
|
||||
<text class="outer_axis_smaller" transform="matrix(1 0 0 1 104 26)">R 255</text>
|
||||
<text class="outer_axis_smaller" transform="matrix(1 0 0 1 19 194)">G 255</text>
|
||||
<text class="outer_axis_smaller" transform="matrix(1 0 0 1 233 180)">B 255</text>
|
||||
</g>
|
||||
</g>
|
||||
|
||||
</g>
|
||||
</svg>
|
||||
|
Before Width: | Height: | Size: 17 KiB |
|
Before Width: | Height: | Size: 278 KiB |
|
Before Width: | Height: | Size: 271 KiB |
|
Before Width: | Height: | Size: 250 KiB |
|
Before Width: | Height: | Size: 297 KiB |
|
Before Width: | Height: | Size: 435 KiB |
|
Before Width: | Height: | Size: 2.4 MiB |
|
Before Width: | Height: | Size: 145 KiB |
|
Before Width: | Height: | Size: 1.8 MiB |
|
|
@ -1,500 +0,0 @@
|
|||
---
|
||||
title: "LED Characterization"
|
||||
date: 2018-05-02T11:18:38+02:00
|
||||
---
|
||||
|
||||
Preface
|
||||
-------
|
||||
|
||||
Recently, I have been working on a `small driver`_ for ambient lighting using 12V LED strips like you can get
|
||||
inexpensively from China. I wanted to be able to just throw one of these somewhere, stick down some LED tape, hook it up
|
||||
to a small transformer and be able to control it through Wifi. When I was writing the firmware, I noticed that when
|
||||
fading between different colors, the colors look *all wrong*! This observation led me down a rabbit hole of color
|
||||
perception and LED peculiarities.
|
||||
|
||||
The idea of the LED driver was that it can be used either with up to eight single-color LED tapes or, much more
|
||||
interesting, with up to two RGB or RGBW (red-green-blue-white) LED tapes. For ambient lighting high color resolution was
|
||||
really important so you could dim it down a lot without flickering. I ended up using the same driver stage I used in the
|
||||
`multichannel LED driver`_ project for its great color resolution and low hardware requirements.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/rgb_cube.svg" alt="An illustration of the RGB color cube.">
|
||||
<figcaption>An illustration of the RGB color cube.
|
||||
<a href="https://commons.wikimedia.org/wiki/File:RGB_color_cube.svg">Picture</a> by
|
||||
<a href="https://commons.wikimedia.org/wiki/User:Maklaan">Maklaan from Wikimedia Commons</a>,
|
||||
<a href="https://creativecommons.org/licenses/by-sa/3.0/">CC-BY-SA 3.0</a>
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
To make setting colors over Wifi more intuitive I implemented support for HSV colors. RGB is fine for communication
|
||||
between computers, but I think HSV is easier to work with when manually inputting colors from the command line. RGB is
|
||||
close to how most monitors, cameras and the human visual apparatus work on a very low level but doesn't match
|
||||
higher-level human color perception very well. When we describe a color we tend to think in terms of "hue" or
|
||||
"brightness", and computing a measure of those from RGB values is not easy.
|
||||
|
||||
Colors and Color Spaces
|
||||
-----------------------
|
||||
|
||||
`Color spaces`_ are a mathematical abstraction of the concept of color. When we say "RGB", most of the time we actually
|
||||
mean `sRGB`_, a standardized notion of how to map three numbers labelled "red", "green" and "blue" onto a perceived
|
||||
color. `HSV`_ is an early attempt to more closely align these numbers with our perception. After HSV, a number of other
|
||||
*perceptual* color spaces such as `XYZ (CIE 1931)`_ and `CIE Lab/LCh`_ were born, further improving this alignment. In
|
||||
this mathematical model, mapping a color from one color space into another color space is just a coordinate
|
||||
transformation.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/hsv_cylinder.png" alt="An illustration of the HSV color space as a cylinder.">
|
||||
<figcaption>An illustration of the HSV color space as a cylinder.
|
||||
<a href="https://commons.wikimedia.org/wiki/File:HSV_color_solid_cylinder.png">Picture</a> by
|
||||
<a href="https://commons.wikimedia.org/wiki/User:SharkD">SharkD from Wikimedia Commons</a>,
|
||||
<a href="https://creativecommons.org/licenses/by-sa/3.0/">CC-BY-SA 3.0</a>
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
CIE 1931 XYZ is much larger than any other color space, which is why it is a good basis to express other color spaces
|
||||
in. In XYZ there are many coordinates that are outside of what the human eye can perceive. Below is an illustration of
|
||||
the sRGB space within XYZ. The wireframe cube is (0,0,0) to (1,1,1) in XYZ. The colorful object in the middle is what
|
||||
of sRGB fits inside XYZ, and the lines extending out from it indicate the space that can be expressed in sRGB but not in
|
||||
XYZ. The fat white curve is a projection of the *monochromatic spectral locus*, that is the curve of points you get in
|
||||
XYZ for pure visible wavelengths.
|
||||
|
||||
As you can see, sRGB is *much* smaller than XYZ or even the part within the monochromatic locus that we can perceive. In
|
||||
particular in the blues and greens we loose *a lot* of colors to sRGB.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<video controls loop>
|
||||
<source src="video/sRGB.mkv" type="video/h264">
|
||||
<source src="video/sRGB.webm" type="video/webm">
|
||||
Your browser does not support the HTML5 video tag.
|
||||
</video>
|
||||
<figcaption>Illustration of the measured sRGB color space within XYZ. The thick, white line is the spectral
|
||||
locus.
|
||||
|
||||
<a href="video/sRGB.mkv">mkv/h264 download</a> /
|
||||
<a href="video/sRGB.webm">webm download</a>
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
The wrong colors I got when fading between colors were caused by this coordinate transformation being askew. Thinking
|
||||
over the problem, there are several sources for imperfections:
|
||||
|
||||
* The LED driver may not be entirely linear. For most modulations such as PWM the brightness will be linear starting
|
||||
from a certain value, but there is probably an offset caused by imperfect edges of the LED current. This offset can be
|
||||
compensated with software calibration. I built a calibration setup for driver linearity in the `multichannel LED
|
||||
driver`_ project. Below are pictures of ringing on the edges of an LED driver's waveform.
|
||||
|
||||
* The red, green and blue channels of the LEDs used on the LED tape are not matched. This skews the RGB color space.
|
||||
In practice, the blue channel of my RGB tape to me *looks* much brighter than the red channel.
|
||||
|
||||
* The precise colors of the red, green and blue channels of the LEDs are unknown. Though the red channel *looks* red, it
|
||||
may be of a slightly different hue compared to the reference red used in `sRGB`_ which would also skew the RGB color
|
||||
space.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<figure class="side-by-side">
|
||||
<img src="images/driver_ringing_strong.jpg" alt="Strong ringing on the LED voltage waveform edge at about
|
||||
100% overshoot during about 70% of the cycle time.">
|
||||
<figcaption>The LED strip being at the end of a couple meters of wire caused extremely bad ringing at high
|
||||
driving frequencies.</figcaption>
|
||||
</figure><figure class="side-by-side">
|
||||
<img src="images/driver_ringing_weak.jpg" alt="Weak ringing on the LED voltage waveform edge at about 30%
|
||||
overshoot during about 20% of the cycle time.">
|
||||
<figcaption>Adding a resistor in front of the MOSFET gate to slow the transition dampened the ringing
|
||||
somewhat, but ultimately it cannot be eliminated entirely.</figcaption>
|
||||
</figure>
|
||||
</figure>
|
||||
|
||||
These last two errors are tricky to compensate. What I needed for that was basically a model of the *perceived* colors
|
||||
of the LED tape's color channels. A way of doing his is to record the spectra of all color channels and then evaluate
|
||||
their respective XYZ coordinates. If all three channels are measured in one go with the same setup the relative
|
||||
magnitudes of the channels in XYZ will be accurate.
|
||||
|
||||
To map any color to the LEDs, the color's XYZ coordinates simply have to be mapped onto the linear coordinate system
|
||||
produced by these three points within XYZ. LEDs are mostly linear in their luminous flux vs. current characteristic so
|
||||
this model will be adequate. The spectral integrals mapping the channels' measured responses to XYZ need only be
|
||||
calculated once and their results can be used as scaling factors thereafter.
|
||||
|
||||
Measuring the spectrum
|
||||
----------------------
|
||||
|
||||
In order to compensate for the cheap LED tape's non-ideal performance I had to measure the LED's red, green and blue
|
||||
channels' spectra. The obvious thing would be to go out and buy a `spectrograph`_, or ask someone to borrow theirs. The
|
||||
former is kind of expensive, and I did not want to wait two weeks for the thing to arrive. The latter I could probably
|
||||
not do every time I got new LED tape. Thus the only choice was to build my own.
|
||||
|
||||
Luckily, building your own spectrometer is really easy. The first thing you need is something that splits incident light
|
||||
into its constituent wavelengths. In professional devices this is called the *`monochromator`_*, since it allows extraction
|
||||
of small color bands from the spectrum. The second thing is some sort of optics that project the incident light onto a
|
||||
screen behind the monochromator. In professional devices lenses or curved mirrors are used. In a simple homebrew job a
|
||||
pinhole as you would use in a `camera obscura`_ does a remarkably nice job.
|
||||
|
||||
For the monochromator component several things could be used. A prism would work, but I did not have any. The
|
||||
alternative is a `diffraction grating`_. Professional gratings are quite specialized pieces of equipment and thus
|
||||
rather expensive. Luckily, there is a common household item that works almost as well: A regular CD or DVD. The
|
||||
microscopic grooves that are used to record data in a CD or DVD work the same as the grooves in a professional
|
||||
diffraction grating.
|
||||
|
||||
Household spectra
|
||||
-----------------
|
||||
|
||||
From this starting point, a few seconds on my favorite search engine yielded an `article by two researchers from the
|
||||
National Science Museum in Tokyo`_ providing a nice blueprint for a simple cardboard-and-DVD construction for use in
|
||||
classrooms. I replicated their device using a DVD and it worked beautifully. Daylight and several types of small LEDs I
|
||||
had around did show the expected spectra. Small red, yellow, green, and blue LEDs showed narrow spectra, daylight one
|
||||
continuous broad one, and white LEDs a continuous broad one with a distinct bright spot in the blue part. The
|
||||
single-color LED spectra are quite narrow since they are determined by the LED's semiconductor's band gap, which is
|
||||
specific to the semiconductor used and is quite precise. White LEDs are in fact a blue LED chip covered with a so-called
|
||||
*phosphor*. This phosphor is not elementary phosphorus but an anorganic compound that absorbs the LED chip's blue light
|
||||
and re-emits a broader spectrum of more yellow-ish wavelengths instead. The final LED spectrum is a superposition of
|
||||
both spectra, with some of the original blue light leaking through the phosphor mixing with the broadband yellow
|
||||
spectrum of the phosphor.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<figure class="side-by-side">
|
||||
<img src="images/spectrograph_step1_parts.jpg">
|
||||
<figcaption>The ingredients. The cup of coffee and Madoka Magica DVD set are essential to the eventual
|
||||
function of the appartus.</figcaption>
|
||||
</figure><figure class="side-by-side">
|
||||
<img src="images/spectrograph_step2.jpg">
|
||||
<figcaption>Step 1: Cut to size and mark down all holes as described in <a
|
||||
href="http://www.candac.ca/candacweb/sites/default/files/BuildaSpectroscope.pdf">the manual</a></figcaption>
|
||||
</figure>
|
||||
<figure class="side-by-side">
|
||||
<img src="images/spectrograph_step3.jpg">
|
||||
<figcaption>Step 2: Cut out all holes</figcaption>
|
||||
</figure><figure class="side-by-side">
|
||||
<img src="images/spectrograph_step4_complete.jpg">
|
||||
<figcaption>The finished result with the back side showing. The viewing window is on the bottom of the other
|
||||
side.</figcaption>
|
||||
</figure>
|
||||
</figure>
|
||||
|
||||
|
||||
Now that I had a spectrograph, I needed a somewhat predictable way of measuring the spectrum it gave me.
|
||||
|
||||
Measuring a spectrum
|
||||
--------------------
|
||||
|
||||
Pointing a camera at the spectrograph would be the obvious thing to do. This produces pretty images but has one critical
|
||||
flaw: I wanted to acquire quantitative measurements of brightness across the spectrum. Since I don't have a precise
|
||||
technical datasheet specifying the spectral response of any of my cameras I can't compare the absolute brightness of
|
||||
different colors on their pictures. Some other sensor was needed.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/daylight_spectrum_dvd.jpg">
|
||||
<figcaption>The daylight spectrum as seen using a DVD as a grating.
|
||||
<a href="https://commons.wikimedia.org/wiki/File:SpectresSolaires-DVD.jpg">Picture</a> by
|
||||
<a href="https://commons.wikimedia.org/wiki/User:Xofc">Xofc from Wikimedia Commons</a>,
|
||||
<a href="https://creativecommons.org/licenses/by-sa/4.0/">CC-BY-SA 4.0</a>
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
|
||||
Measuring light intensity
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
Looking around my lab, I found a bag of `SFH2701`_ visible-light photodiodes. Their
|
||||
datasheet includes their spectral response so I can compensate for that, allowing precise-ish absolute intensity
|
||||
measurements. Just like LEDs, photodiodes are extremely linear across several orders of magnitude. The datasheet of the
|
||||
classic `BPW34`_ photodiode shows that this photodiode's light current is exactly proportional to illuminance over at
|
||||
least three orders of magnitude. The `SFH2701`_ datasheet does not include a similar graph but its performance will be
|
||||
similar. The `SFH2701`_ photodiodes I had at hand were perfect for the job compared to the vintage `BPW34`_ since their
|
||||
active sensing area is really small (0.6mm by 0.6mm) compared to the BPW34 (a whopping 3mm by 3mm). If I were to use a
|
||||
`BPW34`_ I would have to insert some small apterture in front of it so it does not catch too broad a part of the
|
||||
spectrum at once. The `SFH2701`_ is small enough that if I just point it at the projected spectrum directly I will
|
||||
already get only a small part of the spectrum inside its 0.6mm active area.
|
||||
|
||||
To convert the photodiode's tiny photocurrent into a measurable voltage I built another copy of the `transimpedance
|
||||
amplifier`_ circuit I already used in the `multichannel LED driver`_. A `transimpedance amplifier`_ is an
|
||||
amplifiert that produces a large voltage from a small current. The weird name comes from the fact that it works kind of
|
||||
like an amplified resistor (which can be generalized as an *impedance* electrically). Apply a current to a resistor and
|
||||
you get a voltage. A transimpedance amplifiert does the same with the difference that its input always stays at 0V,
|
||||
making it look like an ideal current sink to the connected current source.
|
||||
|
||||
Transimpedance amplifiers are common in optoelectronics to convert small photocurrents to voltages. In this instance I
|
||||
built a very simple circuit with a dampened transimpedance amplifier stage followed by a simple RC filter for noise
|
||||
rejection and a regular non-inverting amplifier using another op-amp from the same chip to further boost the filtered
|
||||
transimpedance amplifier output. I put all the passives setting amplifier response (the gain-setting resistors and the
|
||||
filter resistor and capacitors) on a small removable adapter so I could easily change them if necessary. I put a small
|
||||
trimpot on the virtual ground both amplifers use as a reference so I could trim that if necessary.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/preamp_schematic.jpg" alt="A drawing of the photodiode preamplifier's schematic">
|
||||
<figcaption>The photodiode preamplifier schematic. Schematic drawn with an unlicensed copy of
|
||||
DaveCAD.</figcaption>
|
||||
</figure>
|
||||
|
||||
Following are pictures of the preamplifier board. The connectors on the top-left side are two copies of the analog
|
||||
signal for the ADC and a small panel meter. The SMA connector is used as the photodiode input since coax cables are
|
||||
generally low-leakage and have built-in shielding. The circuit is powered via the micro-USB connector and the analog
|
||||
ground bias voltage can be adjusted using the trimpot.
|
||||
|
||||
For easy replacement, all passives setting gain and frequency response are on a small, pluggable carrier PCB made from a
|
||||
SMD-to-DIP adapter.
|
||||
|
||||
Flying-wire construction is just fine for this low-frequency circuit. In a high-speed photodiode preamp, the
|
||||
transimpedance amplifier circuit would be highly sensitive to stray capacitance, but we're not aiming at high speed
|
||||
here.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<figure class="side-by-side">
|
||||
<img src="images/preamp_front.jpg">
|
||||
<figcaption>The front side of the preamplifier board.</figcaption>
|
||||
</figure><figure class="side-by-side">
|
||||
<img src="images/preamp_back.jpg">
|
||||
<figcaption>The wiring of the photodiode preamp.</figcaption>
|
||||
</figure>
|
||||
</figure>
|
||||
|
||||
Given a way to measure intensity what remains missing is a way to scan a single photodiode across the spectrum.
|
||||
|
||||
Scanning the projection
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
A cheap linear stage can be found in any old CD or DVD drive. These drives use a small linear stage based on a
|
||||
stepper-driven screw to move the laser unit radially. Removing the laser unit and connecting a leftover stepper driver
|
||||
module I was left with a small linear stage with about 45 steps per cm without microstepping enabled. The driver I used
|
||||
was an `A4988`_ module that required at least 8V motor drive voltage. I used a small micro USB-input boost converter
|
||||
module to generate a stable 10V supply for the motor driver, with the USB's 5V rail used as a logic supply for the motor
|
||||
driver.
|
||||
|
||||
The `SFH2701`_ can easily be mounted to the linear stage using a small SMD breakout board glued in place with thin wires
|
||||
connecting it to the transimpedance amplifier. The DVD drive linear stage is not very strong so it is important that
|
||||
this wire does not put too much strain on it.
|
||||
|
||||
Above the photodiode, I mounted a small piece of paper on the linear stage to be used as a projection screen to align
|
||||
the linear stage in front of the spectrometer viewing window. A line on the screen paper points to the photodiode die in
|
||||
parallel to the linear stage allowing precise alignment.
|
||||
|
||||
The whole unit with photodiode preamplifier, linear stage, photodiode and stepper motor driver finally looks like this:
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/electronics_whole.jpg" alt="The complete electronics setup of the spectrograph. In the back
|
||||
there is the DVD drive stepper stage. In front of it, mounted on a piece of wood are a small USB-to-12V
|
||||
switching-regulator module to power the stepper motor in the top left, below on the bottom left is the
|
||||
photodiode preamp and on the right is a breadboard with the stepper driver module and lots of jumper wires
|
||||
interconnecting everything. On the right of the breadboard, a buspirate is attached to interface everything to a
|
||||
computer. On the bottom edge of the piece of wood, two LED panel meters are mounted for readout of the preamp
|
||||
output and the stepper supply voltages.">
|
||||
<figcaption>The complete electronics setup. The buspirate on the right interfaces to a computer and controls the
|
||||
stepper driver and ADC'es the preamp output. The two panel meters show the preamp output and stepper voltage for
|
||||
setup.</figcaption>
|
||||
</figure>
|
||||
|
||||
The projection of the spectrum can be adjusted by moving the light source relative to the entry slot and by moving
|
||||
around the grating DVD.
|
||||
|
||||
The capture process
|
||||
~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
To capture a spectrum, first the light source has to be mounted near the spectrograph's entry slot. The LED tape I
|
||||
tested I just taped face-down directly into it. Next, the grating DVD has to be adjusted to make sure the spectrum
|
||||
covers a sensible part of the photodiode's path. Mostly, this boils down to adjusting the photodiode distance and height
|
||||
to match the vertical extent and wiggling the grating DVD to adjust the projection's horizontal position.
|
||||
|
||||
After the optics are set-up, the photodiode preamplifier has to be adjusted. In my experiments, most LED tape at 5GΩ
|
||||
required a high-ish amplification. The goal in this step is to maximize the peak response of the preamp to be just
|
||||
shy of its VCC rail to make best use of its dynamic range. To adjust the pre-amp, I took several very coarsely-spaced
|
||||
measurements to give me an estimate of the peak while I did not yet know its precise location.
|
||||
|
||||
Since stray daylight totally swamped out the weak projection of the LED's spectrum I shielded the entire setup with a
|
||||
small box made of black cardboard and two black t-shirts on top. This shielding proved adequate for all my measurements
|
||||
but I had to be careful not to accidentially move the DVD that was stuck into the spectrograph with the shielding
|
||||
t-shirts.
|
||||
|
||||
For capturing a single spectrum I wrote a small python script that will automatically move the stepper in adjustable
|
||||
intervals and take two measurements at each point, one with the LED tape off that can be used for offset calibration and
|
||||
one with the LED tape on. All measurements are stored in a sqlite database that can then be accesssed from other
|
||||
scripts.
|
||||
|
||||
I built a small script that shows the progress of the current run and an jupyter notebook for data analysis. The jupyter
|
||||
notebook is capable of live-updating a graph with the in-progress spectrum's data. This was quite useful as a sanity
|
||||
check for when I made some mistake easy to spot in the resulting data.
|
||||
|
||||
After one color channel is captured, the LED tape has to be manually set to the next color and the next measurement can
|
||||
begin.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/raw_plot_cheap_rgb.svg" alt="A plot with three wide peaks, two large peaks on both sides and
|
||||
one smaller one in the middle. The middle one overlaps the two on the sides. The large ones are about 2.5V in
|
||||
amplitude. Overall, the plot is about 300 stepper steps wide with each peak being around 130 steps wide.">
|
||||
<figcaption>A plot of the raw preamp output voltage versus stepper position. From left to right, the three peaks
|
||||
are blue, green and red. Step 0 corresponds to the bottommost stepper position and the shortest wavelength.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
|
||||
Data analysis
|
||||
~~~~~~~~~~~~~
|
||||
|
||||
Data analysis consists of three major steps: Offset- and stray light removal, wavelength and amplitude calibration and
|
||||
color space mapping.
|
||||
|
||||
Offset removal
|
||||
**************
|
||||
The first task is to remove the offset caused by dark current as well as stray light of the LED's bright primary
|
||||
reflection on the DVD. The LED is very bright and only a small part of its light gets reflected by the grating towards
|
||||
the photodiode screen. The remaining part of the light is reflected onto the table in front of the DVD spectrograph.
|
||||
Though I covered all of this with black cardboard, some of that light ultimately gets reflected onto the photodiode.
|
||||
This causes a large offset, in particular in the blue part of the spectrum since in this part the photodiode is closest
|
||||
to the spectrograph's opening.
|
||||
|
||||
The composite offset can be approximated with a second-order polynomial that is fitted to all the data outside of the
|
||||
main peak's area. Since at this point the wavelength of each data point is still unknown this is done with a rough first
|
||||
estimate of the three colors' peaks' locations and widths.
|
||||
|
||||
Wavelength- and amplitude calibration
|
||||
*************************************
|
||||
The photodiode's response is strongly wavelength-dependent. In particular in the blue band, the photodiode's sensitivity
|
||||
gets very poor down to about 20% at the edge to ultraviolet. This effect is strong enough to move the apparent location
|
||||
of the blue peak towards red.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/photodiode_sensitivity.svg" alt="A plot of photodiode sensitivity against wavelength relative
|
||||
to peak sensitivity at 820nm. The sensitivity rises from 20% at 380nm approximately linearly to 80% at 620nm,
|
||||
then the rise rolls off.">
|
||||
<figcaption>A plot of the photodiode's relative sensitivity in the visible spectrum. The sensitivity is
|
||||
normalized against its peak at 820nm.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
The problem is that in order to remove this non-linearity, we would already have to know the wavelength of the measured
|
||||
light. Since I don't, I settled for a two-step process. First, a coarse wavelength calibration is done relative to the
|
||||
red peak and the short-wavelength edge of the blue peak. The photodiode measurements are then sensitivity-corrected
|
||||
using this coarse measurement. Then all three channel peaks are measured in the resulting data and a fine wavelength
|
||||
estimate is produced by a least-squares fit of a linear function. This fine estimate is then used for a second
|
||||
sensitivity correction of all original measurements and the scale is changed from stepper motor step count to
|
||||
wavelength in nanometers.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/processed_plot_cheap_rgb.svg" alt="A plot with three wide peaks, all three of different
|
||||
heights. The leftmost peak is highest at 6nA, the middle peak lowest at 1.6nA and the rightmost peak in between
|
||||
at 4nA. The middle one overlaps the two on the sides. Overall, the plot spans about 300nm on its x axis with
|
||||
each peak being around 100nm wide.">
|
||||
<figcaption>A plot of the processed measurements. From left to right, the three peaks are blue, green and red.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
.. FIXME re-do these measurements, avoiding clipping
|
||||
.. FIXME re-do calibration using CCFL
|
||||
.. FIXME calibration for brightness imbalance due to wedge-shaped projection of spectrum
|
||||
|
||||
Color space mapping
|
||||
*******************
|
||||
Finally, to achieve the objective of measuring the LED tape's channels' precise color coordinates the measured spetra
|
||||
have to be matched against the color spaces' *color matching functions*. The color matching functions describe how
|
||||
strong the color space's idealized *standard observer* would react to light at a particular wavelength. Going from a
|
||||
measured spectrum to color coordinates XYZ works by integrating over the product of the measurement and each color
|
||||
coordinate's color matching function.
|
||||
|
||||
The result are three color coordinates X, Y and Z for each channel R, G and B yielding nine coordinates in total. When
|
||||
written as a matrix conversion between XYZ color space and LED-RGB color space is as simple as multiplying that matrix
|
||||
(or its inverse) and a vector from one of the color spaces.
|
||||
|
||||
In XYZ space, the set of colors that can be produced with this LED tape is described by the `parallelepiped`_ spanned by
|
||||
the three channel's XYZ vectors. In the following figures, you can see a three-dimensional model of the RGB LED's color
|
||||
space (colorful) as well as sRGB (white) for comparison plotted within CIE 1931 XYZ. There is no natural map to scale
|
||||
both so for this illustration the LED color space has been scaled to fit. These figures were made with blender and a few
|
||||
lines of python. The blender project file including all settings and the python script to generate the color space
|
||||
models can be found in the `project repo`_.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<video controls loop>
|
||||
<source src="video/led_within_srgb_scale=1.0.mkv" type="video/h264">
|
||||
<source src="video/led_within_srgb_scale=1.0.webm" type="video/webm">
|
||||
Your browser does not support the HTML5 video tag.
|
||||
</video>
|
||||
<figcaption>Illustration of the measured LED color space scaled to fit within XYZ with sRGB (light gray) for
|
||||
comparison. The thick, white line is the spectral locus.
|
||||
|
||||
<a href="video/led_within_srgb_scale=1.0.mkv">mkv/h264 download</a> /
|
||||
<a href="video/led_within_srgb_scale=1.0.webm">webm download</a>
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
As you can see, the result is pretty disappointing. The LED's color space parallepiped is very narrow, which is because
|
||||
the blue channel is much brighter than the other two channels. An easy fix for this is to scale-up the RGB space and
|
||||
drop any values outside XYZ. The scaling factor is a trade-off between color space coverage and brightness. You can
|
||||
produce the most colors when you clip all channels to brightness of the weakest channel (green in this case), but that
|
||||
will make the result very dim. Scaling brightness like that stretches the RGB parallelepiped along its major axis. Up to
|
||||
a point the number of possible colors (the gamut) increases at expense of maximum brightness. When the parallelepiped is
|
||||
stretched far enought for all three channel vectors to be outside the 1,1,1 XYZ-cube, maximum brightness continues to
|
||||
decrease but the gamut stays constant. I don't know a simple scientific way to solve this problem, so I just played
|
||||
around with a couple of factors and settled on 2.5 as a reasonable compromise. Below is an illustration.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<video controls loop>
|
||||
<source src="video/led_within_srgb_fancy_camera_path_scale=2.5.mkv" type="video/h264">
|
||||
<source src="video/led_within_srgb_fancy_camera_path_scale=2.5.webm" type="video/webm">
|
||||
Your browser does not support the HTML5 video tag.
|
||||
</video>
|
||||
<figcaption>Illustration of the measured LED color space at scale factor 2.5 within XYZ with sRGB (light gray)
|
||||
for comparison. The thick, white line is the spectral locus.
|
||||
|
||||
<a href="video/led_within_srgb_fancy_camera_path_scale=2.5.mkv">mkv/h264 download</a> /
|
||||
<a href="video/led_within_srgb_fancy_camera_path_scale=2.5.webm">webm download</a>
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
Firmware implementation
|
||||
-----------------------
|
||||
In the end, the above measurements yield two matrices: One for mapping XYZ to RGB, and one for mapping RGB to XYZ. Of
|
||||
the several versions of CIE XYZ I chose the CIE 1931 XYZ color space as a basis for the firmware because it is most
|
||||
popular. Mapping a color coordinate in one color space to the other is as simple as performing nine floating-point
|
||||
multiplications and six additions. Mapping Lab or Lch to RGB is done by first mapping Lab/Lch to XYZ, then XYZ to RGB.
|
||||
Lab to XYZ is somewhat complex since it requires a floating-point power for gamma correction, but any self-respecting
|
||||
libc will have one of those so this is still no problem. Lch also requires floating-point sine and cosine functions, but
|
||||
these should still be no problem on most hardware.
|
||||
|
||||
My implementation of these conversions in the ESP8266 firmware of my `Wifi LED driver`_ can be found `on Github`_. You
|
||||
can view the Jupyter notebook most of the analysis above `here <http://nbviewer.jupyter.org/github/jaseg/led_drv/blob/master/doc/Spectrum%20Measurement.ipynb>`__.
|
||||
|
||||
.. _`on Github`: https://github.com/jaseg/esp_led_drv/blob/master/user/led_controller.c
|
||||
.. _`project repo`: https://github.com/jaseg/led_drv
|
||||
.. _`Wifi LED driver`: {{<ref "posts/wifi-led-driver/index.rst">}}
|
||||
.. _`small driver`: {{<ref "posts/wifi-led-driver/index.rst">}}
|
||||
.. _`multichannel LED driver`: {{<ref "posts/multichannel-led-driver/index.rst">}}
|
||||
.. _`sRGB`: https://en.wikipedia.org/wiki/SRGB
|
||||
.. _`CC BY-SA 3.0`: https://creativecommons.org/licenses/by-sa/3.0
|
||||
.. _`Color spaces`: https://en.wikipedia.org/wiki/Color_space
|
||||
.. _`HSV`: https://en.wikipedia.org/wiki/HSL_and_HSV
|
||||
.. _`CIE Lab/LCh`: https://en.wikipedia.org/wiki/Lab_color_space
|
||||
.. _`XYZ (CIE 1931)`: https://en.wikipedia.org/wiki/CIE_1931_color_space
|
||||
.. _`camera obscura`: https://en.wikipedia.org/wiki/Pinhole_camera
|
||||
.. _`article by two researchers from the National Science Museum in Tokyo`: http://www.candac.ca/candacweb/sites/default/files/BuildaSpectroscope.pdf
|
||||
.. _`spectrograph`: https://en.wikipedia.org/wiki/Ultraviolet%E2%80%93visible_spectroscopy
|
||||
.. _`monochromator`: https://en.wikipedia.org/wiki/Monochromator
|
||||
.. _`diffraction grating`: https://en.wikipedia.org/wiki/Diffraction_grating
|
||||
.. _`SFH2701`: https://dammedia.osram.info/media/resource/hires/osram-dam-2495903/SFH%202701.pdf
|
||||
.. _`BPW34`: http://www.vishay.com/docs/81521/bpw34.pdf
|
||||
.. _`transimpedance amplifier`: https://en.wikipedia.org/wiki/Transimpedance_amplifier
|
||||
.. _`A4988`: https://www.pololu.com/file/0J450/A4988.pdf
|
||||
.. _`parallelepiped`: https://en.wikipedia.org/wiki/Parallelepiped
|
||||
|
Before Width: | Height: | Size: 69 KiB |
|
Before Width: | Height: | Size: 58 KiB |
|
Before Width: | Height: | Size: 1.6 MiB |
|
|
@ -1,765 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8" standalone="no"?>
|
||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN"
|
||||
"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||
<!-- Created with matplotlib (http://matplotlib.org/) -->
|
||||
<svg height="288pt" version="1.1" viewBox="0 0 432 288" width="432pt" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||
<defs>
|
||||
<style type="text/css">
|
||||
*{stroke-linecap:butt;stroke-linejoin:round;}
|
||||
</style>
|
||||
</defs>
|
||||
<g id="figure_1">
|
||||
<g id="patch_1">
|
||||
<path d="M 0 288
|
||||
L 432 288
|
||||
L 432 0
|
||||
L 0 0
|
||||
z
|
||||
" style="fill:none;opacity:0;"/>
|
||||
</g>
|
||||
<g id="axes_1">
|
||||
<g id="patch_2">
|
||||
<path d="M 54 256.32
|
||||
L 388.8 256.32
|
||||
L 388.8 34.56
|
||||
L 54 34.56
|
||||
z
|
||||
" style="fill:#ffffff;"/>
|
||||
</g>
|
||||
<g id="matplotlib.axis_1">
|
||||
<g id="xtick_1">
|
||||
<g id="line2d_1">
|
||||
<path clip-path="url(#pd6435b8b02)" d="M 54 256.32
|
||||
L 54 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_2">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L 0 3.5
|
||||
" id="m1925175e85" style="stroke:#01769d;stroke-width:0.8;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m1925175e85" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_1">
|
||||
<!-- ${10^{0}}$ -->
|
||||
<defs>
|
||||
<path d="M 12.40625 8.296875
|
||||
L 28.515625 8.296875
|
||||
L 28.515625 63.921875
|
||||
L 10.984375 60.40625
|
||||
L 10.984375 69.390625
|
||||
L 28.421875 72.90625
|
||||
L 38.28125 72.90625
|
||||
L 38.28125 8.296875
|
||||
L 54.390625 8.296875
|
||||
L 54.390625 0
|
||||
L 12.40625 0
|
||||
z
|
||||
" id="DejaVuSans-31"/>
|
||||
<path d="M 31.78125 66.40625
|
||||
Q 24.171875 66.40625 20.328125 58.90625
|
||||
Q 16.5 51.421875 16.5 36.375
|
||||
Q 16.5 21.390625 20.328125 13.890625
|
||||
Q 24.171875 6.390625 31.78125 6.390625
|
||||
Q 39.453125 6.390625 43.28125 13.890625
|
||||
Q 47.125 21.390625 47.125 36.375
|
||||
Q 47.125 51.421875 43.28125 58.90625
|
||||
Q 39.453125 66.40625 31.78125 66.40625
|
||||
M 31.78125 74.21875
|
||||
Q 44.046875 74.21875 50.515625 64.515625
|
||||
Q 56.984375 54.828125 56.984375 36.375
|
||||
Q 56.984375 17.96875 50.515625 8.265625
|
||||
Q 44.046875 -1.421875 31.78125 -1.421875
|
||||
Q 19.53125 -1.421875 13.0625 8.265625
|
||||
Q 6.59375 17.96875 6.59375 36.375
|
||||
Q 6.59375 54.828125 13.0625 64.515625
|
||||
Q 19.53125 74.21875 31.78125 74.21875
|
||||
" id="DejaVuSans-30"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(45.2 270.918437)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.765625)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.765625)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 39.046875)scale(0.7)" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_2">
|
||||
<g id="line2d_3">
|
||||
<path clip-path="url(#pd6435b8b02)" d="M 177.575725 256.32
|
||||
L 177.575725 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_4">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="177.575725" xlink:href="#m1925175e85" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_2">
|
||||
<!-- ${10^{1}}$ -->
|
||||
<g style="fill:#01769d;" transform="translate(168.775725 270.918437)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.684375)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.684375)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 38.965625)scale(0.7)" xlink:href="#DejaVuSans-31"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_3">
|
||||
<g id="line2d_5">
|
||||
<path clip-path="url(#pd6435b8b02)" d="M 301.15145 256.32
|
||||
L 301.15145 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_6">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="301.15145" xlink:href="#m1925175e85" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_3">
|
||||
<!-- ${10^{2}}$ -->
|
||||
<defs>
|
||||
<path d="M 19.1875 8.296875
|
||||
L 53.609375 8.296875
|
||||
L 53.609375 0
|
||||
L 7.328125 0
|
||||
L 7.328125 8.296875
|
||||
Q 12.9375 14.109375 22.625 23.890625
|
||||
Q 32.328125 33.6875 34.8125 36.53125
|
||||
Q 39.546875 41.84375 41.421875 45.53125
|
||||
Q 43.3125 49.21875 43.3125 52.78125
|
||||
Q 43.3125 58.59375 39.234375 62.25
|
||||
Q 35.15625 65.921875 28.609375 65.921875
|
||||
Q 23.96875 65.921875 18.8125 64.3125
|
||||
Q 13.671875 62.703125 7.8125 59.421875
|
||||
L 7.8125 69.390625
|
||||
Q 13.765625 71.78125 18.9375 73
|
||||
Q 24.125 74.21875 28.421875 74.21875
|
||||
Q 39.75 74.21875 46.484375 68.546875
|
||||
Q 53.21875 62.890625 53.21875 53.421875
|
||||
Q 53.21875 48.921875 51.53125 44.890625
|
||||
Q 49.859375 40.875 45.40625 35.40625
|
||||
Q 44.1875 33.984375 37.640625 27.21875
|
||||
Q 31.109375 20.453125 19.1875 8.296875
|
||||
" id="DejaVuSans-32"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(292.35145 270.918437)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.765625)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.765625)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 39.046875)scale(0.7)" xlink:href="#DejaVuSans-32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_4">
|
||||
<g id="line2d_7">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L 0 2
|
||||
" id="m2cc6d12e8a" style="stroke:#01769d;stroke-width:0.6;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="91.2" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_5">
|
||||
<g id="line2d_8">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="112.960605" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_6">
|
||||
<g id="line2d_9">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="128.4" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_7">
|
||||
<g id="line2d_10">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="140.375725" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_8">
|
||||
<g id="line2d_11">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="150.160605" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_9">
|
||||
<g id="line2d_12">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="158.433603" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_10">
|
||||
<g id="line2d_13">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="165.6" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_11">
|
||||
<g id="line2d_14">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="171.92121" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_12">
|
||||
<g id="line2d_15">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="214.775725" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_13">
|
||||
<g id="line2d_16">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="236.53633" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_14">
|
||||
<g id="line2d_17">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="251.975725" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_15">
|
||||
<g id="line2d_18">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="263.95145" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_16">
|
||||
<g id="line2d_19">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="273.73633" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_17">
|
||||
<g id="line2d_20">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="282.009328" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_18">
|
||||
<g id="line2d_21">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="289.175725" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_19">
|
||||
<g id="line2d_22">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="295.496935" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_20">
|
||||
<g id="line2d_23">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="338.35145" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_21">
|
||||
<g id="line2d_24">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="360.112055" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_22">
|
||||
<g id="line2d_25">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="375.55145" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_23">
|
||||
<g id="line2d_26">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="387.527175" xlink:href="#m2cc6d12e8a" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="matplotlib.axis_2">
|
||||
<g id="ytick_1">
|
||||
<g id="line2d_27">
|
||||
<path clip-path="url(#pd6435b8b02)" d="M 54 256.114166
|
||||
L 388.8 256.114166
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_28">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L -3.5 0
|
||||
" id="m09fd9014c3" style="stroke:#01769d;stroke-width:0.8;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m09fd9014c3" y="256.114166"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_4">
|
||||
<!-- ${10^{-2}}$ -->
|
||||
<defs>
|
||||
<path d="M 10.59375 35.5
|
||||
L 73.1875 35.5
|
||||
L 73.1875 27.203125
|
||||
L 10.59375 27.203125
|
||||
z
|
||||
" id="DejaVuSans-2212"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(23.5 259.913384)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.765625)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.765625)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 39.046875)scale(0.7)" xlink:href="#DejaVuSans-2212"/>
|
||||
<use transform="translate(186.855469 39.046875)scale(0.7)" xlink:href="#DejaVuSans-32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_2">
|
||||
<g id="line2d_29">
|
||||
<path clip-path="url(#pd6435b8b02)" d="M 54 183.830454
|
||||
L 388.8 183.830454
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_30">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m09fd9014c3" y="183.830454"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_5">
|
||||
<!-- ${10^{-1}}$ -->
|
||||
<g style="fill:#01769d;" transform="translate(23.5 187.629673)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.684375)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.684375)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 38.965625)scale(0.7)" xlink:href="#DejaVuSans-2212"/>
|
||||
<use transform="translate(186.855469 38.965625)scale(0.7)" xlink:href="#DejaVuSans-31"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_3">
|
||||
<g id="line2d_31">
|
||||
<path clip-path="url(#pd6435b8b02)" d="M 54 111.546743
|
||||
L 388.8 111.546743
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_32">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m09fd9014c3" y="111.546743"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_6">
|
||||
<!-- ${10^{0}}$ -->
|
||||
<g style="fill:#01769d;" transform="translate(29.4 115.345962)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.765625)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.765625)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 39.046875)scale(0.7)" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_4">
|
||||
<g id="line2d_33">
|
||||
<path clip-path="url(#pd6435b8b02)" d="M 54 39.263032
|
||||
L 388.8 39.263032
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_34">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m09fd9014c3" y="39.263032"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_7">
|
||||
<!-- ${10^{1}}$ -->
|
||||
<g style="fill:#01769d;" transform="translate(29.4 43.062251)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.684375)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.684375)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 38.965625)scale(0.7)" xlink:href="#DejaVuSans-31"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_5">
|
||||
<g id="line2d_35">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L -2 0
|
||||
" id="mcde38ab5e9" style="stroke:#01769d;stroke-width:0.6;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="234.3546"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_6">
|
||||
<g id="line2d_36">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="221.626071"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_7">
|
||||
<g id="line2d_37">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="212.595035"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_8">
|
||||
<g id="line2d_38">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="205.59002"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_9">
|
||||
<g id="line2d_39">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="199.866505"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_10">
|
||||
<g id="line2d_40">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="195.027343"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_11">
|
||||
<g id="line2d_41">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="190.83547"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_12">
|
||||
<g id="line2d_42">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="187.137976"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_13">
|
||||
<g id="line2d_43">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="162.070889"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_14">
|
||||
<g id="line2d_44">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="149.34236"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_15">
|
||||
<g id="line2d_45">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="140.311324"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_16">
|
||||
<g id="line2d_46">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="133.306309"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_17">
|
||||
<g id="line2d_47">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="127.582794"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_18">
|
||||
<g id="line2d_48">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="122.743632"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_19">
|
||||
<g id="line2d_49">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="118.551759"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_20">
|
||||
<g id="line2d_50">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="114.854265"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_21">
|
||||
<g id="line2d_51">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="89.787178"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_22">
|
||||
<g id="line2d_52">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="77.058649"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_23">
|
||||
<g id="line2d_53">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="68.027613"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_24">
|
||||
<g id="line2d_54">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="61.022598"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_25">
|
||||
<g id="line2d_55">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="55.299083"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_26">
|
||||
<g id="line2d_56">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="50.459921"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_27">
|
||||
<g id="line2d_57">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="46.268048"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_28">
|
||||
<g id="line2d_58">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#mcde38ab5e9" y="42.570554"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="line2d_59">
|
||||
<path clip-path="url(#pd6435b8b02)" d="M -1 278.211573
|
||||
L 54 246.24
|
||||
L 91.2 218.603412
|
||||
L 112.960605 207.714295
|
||||
L 128.4 197.229035
|
||||
L 140.375725 191.248204
|
||||
L 150.160605 184.371583
|
||||
L 158.433603 180.276352
|
||||
L 165.6 174.141495
|
||||
L 171.92121 171.132596
|
||||
L 177.575725 167.322786
|
||||
L 182.690856 164.879122
|
||||
L 187.360605 161.849412
|
||||
L 191.656358 159.784175
|
||||
L 195.633603 157.082085
|
||||
L 199.33633 155.299733
|
||||
L 202.8 152.474945
|
||||
L 206.053618 150.929999
|
||||
L 209.12121 148.870742
|
||||
L 212.022904 147.48974
|
||||
L 214.775725 145.713065
|
||||
L 217.394208 144.461619
|
||||
L 219.890856 142.775715
|
||||
L 222.276505 141.634047
|
||||
L 224.560605 139.71538
|
||||
L 226.75145 138.678024
|
||||
L 228.856358 137.26953
|
||||
L 230.881815 136.308751
|
||||
L 232.833603 135.053978
|
||||
L 234.716893 134.157744
|
||||
L 236.53633 132.934508
|
||||
L 238.296103 132.096012
|
||||
L 240 129.910697
|
||||
L 241.651461 129.148272
|
||||
L 243.253618 128.102487
|
||||
L 244.809328 127.382252
|
||||
L 246.32121 126.433394
|
||||
L 247.791665 125.75005
|
||||
L 249.222904 124.80997
|
||||
L 250.616963 124.160712
|
||||
L 251.975725 123.04673
|
||||
L 253.300935 122.43259
|
||||
L 254.594208 121.585525
|
||||
L 255.857049 120.999056
|
||||
L 257.090856 120.222645
|
||||
L 258.296935 119.660871
|
||||
L 259.476505 118.884512
|
||||
L 260.630705 118.345981
|
||||
L 261.760605 117.448109
|
||||
L 262.867206 116.933468
|
||||
L 263.95145 116.220985
|
||||
L 265.014223 115.725919
|
||||
L 266.056358 115.068312
|
||||
L 267.078641 114.590958
|
||||
L 268.081815 113.92917
|
||||
L 269.066581 113.468704
|
||||
L 270.033603 112.672236
|
||||
L 270.983509 112.229715
|
||||
L 271.916893 111.615413
|
||||
L 272.834321 111.187442
|
||||
L 273.73633 110.617556
|
||||
L 274.623429 110.202887
|
||||
L 275.496103 109.626643
|
||||
L 276.354813 109.224777
|
||||
L 277.2 109.295322
|
||||
L 278.032083 108.897648
|
||||
L 278.851461 108.344666
|
||||
L 279.658518 107.958782
|
||||
L 282.009328 106.547154
|
||||
L 282.770593 106.18262
|
||||
L 283.52121 105.549474
|
||||
L 284.261474 105.19628
|
||||
L 284.991665 104.704322
|
||||
L 287.124459 103.565923
|
||||
L 287.816963 103.098678
|
||||
L 288.500644 102.771871
|
||||
L 289.175725 102.222111
|
||||
L 289.84242 101.904257
|
||||
L 291.151467 101.150656
|
||||
L 293.057049 100.009312
|
||||
L 293.677498 99.712991
|
||||
L 294.290856 99.196798
|
||||
L 296.089961 98.222269
|
||||
L 297.830705 97.180912
|
||||
L 298.398629 96.910011
|
||||
L 298.960605 96.180743
|
||||
L 300.612066 95.294238
|
||||
L 302.214223 94.34423
|
||||
L 302.73782 94.096643
|
||||
L 303.256358 93.664432
|
||||
L 304.782572 92.84533
|
||||
L 306.266581 91.965625
|
||||
L 306.75227 91.736038
|
||||
L 307.233603 91.348225
|
||||
L 308.65223 90.586682
|
||||
L 310.034321 89.767258
|
||||
L 310.487221 89.553146
|
||||
L 310.93633 89.178828
|
||||
L 312.26154 88.467563
|
||||
L 313.97907 87.500464
|
||||
L 314.4 88.186908
|
||||
L 315.643336 87.497525
|
||||
L 318.43711 85.759568
|
||||
L 319.59131 85.131863
|
||||
L 321.461474 83.944378
|
||||
L 322.553069 83.351609
|
||||
L 324.324459 82.218496
|
||||
L 325.359892 81.657155
|
||||
L 326.039246 81.259501
|
||||
L 326.375725 80.814516
|
||||
L 328.027186 79.896859
|
||||
L 329.629343 78.879938
|
||||
L 331.185053 78.016377
|
||||
L 331.794926 77.619082
|
||||
L 333.289961 76.730618
|
||||
L 334.16739 76.201159
|
||||
L 335.598629 75.351406
|
||||
L 335.880352 75.215965
|
||||
L 336.160605 75.239824
|
||||
L 337.540328 74.46966
|
||||
L 339.414223 73.311292
|
||||
L 340.197715 72.88031
|
||||
L 340.969933 72.367919
|
||||
L 342.232774 71.664338
|
||||
L 342.976457 71.164274
|
||||
L 344.193476 70.486875
|
||||
L 344.433603 70.170499
|
||||
L 345.618381 69.513989
|
||||
L 347.234321 68.522138
|
||||
L 348.359483 67.86112
|
||||
L 349.46154 67.207574
|
||||
L 350.54142 66.567357
|
||||
L 351.389948 66.082245
|
||||
L 351.6 66.720126
|
||||
L 353.047786 65.888213
|
||||
L 353.656507 65.471636
|
||||
L 354.655944 64.905575
|
||||
L 355.246793 64.507405
|
||||
L 356.217313 63.958318
|
||||
L 356.79131 63.566425
|
||||
L 357.734538 63.033416
|
||||
L 358.107235 62.692817
|
||||
L 359.753069 61.679732
|
||||
L 360.646072 61.177566
|
||||
L 361.174828 60.823731
|
||||
L 362.044673 60.334968
|
||||
L 362.559892 59.985558
|
||||
L 364.737062 58.860761
|
||||
L 365.227186 58.527296
|
||||
L 366.353698 57.861658
|
||||
L 367.76817 56.991094
|
||||
L 368.538173 56.558119
|
||||
L 368.843107 56.280724
|
||||
L 370.489961 55.283376
|
||||
L 371.512244 54.678333
|
||||
L 373.220662 53.661782
|
||||
L 373.360605 53.893698
|
||||
L 374.330122 53.338366
|
||||
L 374.876369 52.991801
|
||||
L 377.13782 51.612576
|
||||
L 377.527192 51.395928
|
||||
L 377.785213 51.160437
|
||||
L 379.182572 50.311178
|
||||
L 380.299408 49.652865
|
||||
L 381.513674 48.921946
|
||||
L 381.753265 48.873818
|
||||
L 383.285064 47.946437
|
||||
L 384.320497 47.33543
|
||||
L 384.887221 46.974453
|
||||
L 385.448023 46.584031
|
||||
L 386.879261 45.720998
|
||||
L 387.848223 45.151429
|
||||
L 388.37907 44.814583
|
||||
L 388.695077 44.64
|
||||
L 388.695077 44.64
|
||||
" style="fill:none;stroke:#fe3ea0;stroke-linecap:square;stroke-width:1.5;"/>
|
||||
</g>
|
||||
<g id="patch_3">
|
||||
<path d="M 54 256.32
|
||||
L 54 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-linecap:square;stroke-linejoin:miter;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="patch_4">
|
||||
<path d="M 54 256.32
|
||||
L 388.8 256.32
|
||||
" style="fill:none;stroke:#08bdf9;stroke-linecap:square;stroke-linejoin:miter;stroke-width:0.8;"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<defs>
|
||||
<clipPath id="pd6435b8b02">
|
||||
<rect height="221.76" width="334.8" x="54" y="34.56"/>
|
||||
</clipPath>
|
||||
</defs>
|
||||
</svg>
|
||||
|
Before Width: | Height: | Size: 22 KiB |
|
|
@ -1,937 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8" standalone="no"?>
|
||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN"
|
||||
"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||
<!-- Created with matplotlib (http://matplotlib.org/) -->
|
||||
<svg height="288pt" version="1.1" viewBox="0 0 432 288" width="432pt" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||
<defs>
|
||||
<style type="text/css">
|
||||
*{stroke-linecap:butt;stroke-linejoin:round;}
|
||||
</style>
|
||||
</defs>
|
||||
<g id="figure_1">
|
||||
<g id="patch_1">
|
||||
<path d="M 0 288
|
||||
L 432 288
|
||||
L 432 0
|
||||
L 0 0
|
||||
z
|
||||
" style="fill:none;opacity:0;"/>
|
||||
</g>
|
||||
<g id="axes_1">
|
||||
<g id="patch_2">
|
||||
<path d="M 54 256.32
|
||||
L 388.8 256.32
|
||||
L 388.8 34.56
|
||||
L 54 34.56
|
||||
z
|
||||
" style="fill:#ffffff;"/>
|
||||
</g>
|
||||
<g id="LineCollection_1">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 60.981355 256.32
|
||||
L 60.981355 256.32
|
||||
" style="fill:none;stroke:#ffd2e9;stroke-width:1.5;"/>
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 106.91039 238.778387
|
||||
L 106.91039 238.778387
|
||||
" style="fill:none;stroke:#ffd2e9;stroke-width:1.5;"/>
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 152.839424 220.494693
|
||||
L 152.839424 217.757228
|
||||
" style="fill:none;stroke:#ffd2e9;stroke-width:1.5;"/>
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 198.768458 189.701028
|
||||
L 198.768458 189.701028
|
||||
" style="fill:none;stroke:#ffd2e9;stroke-width:1.5;"/>
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 244.697493 156.978572
|
||||
L 244.697493 156.978572
|
||||
" style="fill:none;stroke:#ffd2e9;stroke-width:1.5;"/>
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 290.626527 119.358916
|
||||
L 290.626527 119.358916
|
||||
" style="fill:none;stroke:#ffd2e9;stroke-width:1.5;"/>
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 336.555562 80.435304
|
||||
L 336.555562 80.435304
|
||||
" style="fill:none;stroke:#ffd2e9;stroke-width:1.5;"/>
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 382.484596 40.266375
|
||||
L 382.484596 40.266375
|
||||
" style="fill:none;stroke:#ffd2e9;stroke-width:1.5;"/>
|
||||
</g>
|
||||
<g id="line2d_1">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 60.981355 256.32
|
||||
L 106.91039 238.778387
|
||||
L 152.839424 219.110317
|
||||
L 198.768458 189.701028
|
||||
L 244.697493 156.978572
|
||||
L 290.626527 119.358916
|
||||
L 336.555562 80.435304
|
||||
L 382.484596 40.266375
|
||||
" style="fill:none;stroke:#ffd2e9;stroke-linecap:square;stroke-width:1.5;"/>
|
||||
</g>
|
||||
<g id="matplotlib.axis_1">
|
||||
<g id="xtick_1">
|
||||
<g id="line2d_2">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 60.981355 256.32
|
||||
L 60.981355 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_3">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L 0 3.5
|
||||
" id="m894b1b7ca2" style="stroke:#01769d;stroke-width:0.8;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="60.981355" xlink:href="#m894b1b7ca2" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_1">
|
||||
<!-- 0 -->
|
||||
<defs>
|
||||
<path d="M 31.78125 66.40625
|
||||
Q 24.171875 66.40625 20.328125 58.90625
|
||||
Q 16.5 51.421875 16.5 36.375
|
||||
Q 16.5 21.390625 20.328125 13.890625
|
||||
Q 24.171875 6.390625 31.78125 6.390625
|
||||
Q 39.453125 6.390625 43.28125 13.890625
|
||||
Q 47.125 21.390625 47.125 36.375
|
||||
Q 47.125 51.421875 43.28125 58.90625
|
||||
Q 39.453125 66.40625 31.78125 66.40625
|
||||
M 31.78125 74.21875
|
||||
Q 44.046875 74.21875 50.515625 64.515625
|
||||
Q 56.984375 54.828125 56.984375 36.375
|
||||
Q 56.984375 17.96875 50.515625 8.265625
|
||||
Q 44.046875 -1.421875 31.78125 -1.421875
|
||||
Q 19.53125 -1.421875 13.0625 8.265625
|
||||
Q 6.59375 17.96875 6.59375 36.375
|
||||
Q 6.59375 54.828125 13.0625 64.515625
|
||||
Q 19.53125 74.21875 31.78125 74.21875
|
||||
" id="DejaVuSans-30"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(57.800105 270.918437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_2">
|
||||
<g id="line2d_4">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 106.91039 256.32
|
||||
L 106.91039 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_5">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="106.91039" xlink:href="#m894b1b7ca2" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_2">
|
||||
<!-- 1 -->
|
||||
<defs>
|
||||
<path d="M 12.40625 8.296875
|
||||
L 28.515625 8.296875
|
||||
L 28.515625 63.921875
|
||||
L 10.984375 60.40625
|
||||
L 10.984375 69.390625
|
||||
L 28.421875 72.90625
|
||||
L 38.28125 72.90625
|
||||
L 38.28125 8.296875
|
||||
L 54.390625 8.296875
|
||||
L 54.390625 0
|
||||
L 12.40625 0
|
||||
z
|
||||
" id="DejaVuSans-31"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(103.72914 270.918437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-31"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_3">
|
||||
<g id="line2d_6">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 152.839424 256.32
|
||||
L 152.839424 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_7">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="152.839424" xlink:href="#m894b1b7ca2" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_3">
|
||||
<!-- 2 -->
|
||||
<defs>
|
||||
<path d="M 19.1875 8.296875
|
||||
L 53.609375 8.296875
|
||||
L 53.609375 0
|
||||
L 7.328125 0
|
||||
L 7.328125 8.296875
|
||||
Q 12.9375 14.109375 22.625 23.890625
|
||||
Q 32.328125 33.6875 34.8125 36.53125
|
||||
Q 39.546875 41.84375 41.421875 45.53125
|
||||
Q 43.3125 49.21875 43.3125 52.78125
|
||||
Q 43.3125 58.59375 39.234375 62.25
|
||||
Q 35.15625 65.921875 28.609375 65.921875
|
||||
Q 23.96875 65.921875 18.8125 64.3125
|
||||
Q 13.671875 62.703125 7.8125 59.421875
|
||||
L 7.8125 69.390625
|
||||
Q 13.765625 71.78125 18.9375 73
|
||||
Q 24.125 74.21875 28.421875 74.21875
|
||||
Q 39.75 74.21875 46.484375 68.546875
|
||||
Q 53.21875 62.890625 53.21875 53.421875
|
||||
Q 53.21875 48.921875 51.53125 44.890625
|
||||
Q 49.859375 40.875 45.40625 35.40625
|
||||
Q 44.1875 33.984375 37.640625 27.21875
|
||||
Q 31.109375 20.453125 19.1875 8.296875
|
||||
" id="DejaVuSans-32"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(149.658174 270.918437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_4">
|
||||
<g id="line2d_8">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 198.768458 256.32
|
||||
L 198.768458 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_9">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="198.768458" xlink:href="#m894b1b7ca2" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_4">
|
||||
<!-- 3 -->
|
||||
<defs>
|
||||
<path d="M 40.578125 39.3125
|
||||
Q 47.65625 37.796875 51.625 33
|
||||
Q 55.609375 28.21875 55.609375 21.1875
|
||||
Q 55.609375 10.40625 48.1875 4.484375
|
||||
Q 40.765625 -1.421875 27.09375 -1.421875
|
||||
Q 22.515625 -1.421875 17.65625 -0.515625
|
||||
Q 12.796875 0.390625 7.625 2.203125
|
||||
L 7.625 11.71875
|
||||
Q 11.71875 9.328125 16.59375 8.109375
|
||||
Q 21.484375 6.890625 26.8125 6.890625
|
||||
Q 36.078125 6.890625 40.9375 10.546875
|
||||
Q 45.796875 14.203125 45.796875 21.1875
|
||||
Q 45.796875 27.640625 41.28125 31.265625
|
||||
Q 36.765625 34.90625 28.71875 34.90625
|
||||
L 20.21875 34.90625
|
||||
L 20.21875 43.015625
|
||||
L 29.109375 43.015625
|
||||
Q 36.375 43.015625 40.234375 45.921875
|
||||
Q 44.09375 48.828125 44.09375 54.296875
|
||||
Q 44.09375 59.90625 40.109375 62.90625
|
||||
Q 36.140625 65.921875 28.71875 65.921875
|
||||
Q 24.65625 65.921875 20.015625 65.03125
|
||||
Q 15.375 64.15625 9.8125 62.3125
|
||||
L 9.8125 71.09375
|
||||
Q 15.4375 72.65625 20.34375 73.4375
|
||||
Q 25.25 74.21875 29.59375 74.21875
|
||||
Q 40.828125 74.21875 47.359375 69.109375
|
||||
Q 53.90625 64.015625 53.90625 55.328125
|
||||
Q 53.90625 49.265625 50.4375 45.09375
|
||||
Q 46.96875 40.921875 40.578125 39.3125
|
||||
" id="DejaVuSans-33"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(195.587208 270.918437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-33"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_5">
|
||||
<g id="line2d_10">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 244.697493 256.32
|
||||
L 244.697493 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_11">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="244.697493" xlink:href="#m894b1b7ca2" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_5">
|
||||
<!-- 4 -->
|
||||
<defs>
|
||||
<path d="M 37.796875 64.3125
|
||||
L 12.890625 25.390625
|
||||
L 37.796875 25.390625
|
||||
z
|
||||
M 35.203125 72.90625
|
||||
L 47.609375 72.90625
|
||||
L 47.609375 25.390625
|
||||
L 58.015625 25.390625
|
||||
L 58.015625 17.1875
|
||||
L 47.609375 17.1875
|
||||
L 47.609375 0
|
||||
L 37.796875 0
|
||||
L 37.796875 17.1875
|
||||
L 4.890625 17.1875
|
||||
L 4.890625 26.703125
|
||||
z
|
||||
" id="DejaVuSans-34"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(241.516243 270.918437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-34"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_6">
|
||||
<g id="line2d_12">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 290.626527 256.32
|
||||
L 290.626527 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_13">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="290.626527" xlink:href="#m894b1b7ca2" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_6">
|
||||
<!-- 5 -->
|
||||
<defs>
|
||||
<path d="M 10.796875 72.90625
|
||||
L 49.515625 72.90625
|
||||
L 49.515625 64.59375
|
||||
L 19.828125 64.59375
|
||||
L 19.828125 46.734375
|
||||
Q 21.96875 47.46875 24.109375 47.828125
|
||||
Q 26.265625 48.1875 28.421875 48.1875
|
||||
Q 40.625 48.1875 47.75 41.5
|
||||
Q 54.890625 34.8125 54.890625 23.390625
|
||||
Q 54.890625 11.625 47.5625 5.09375
|
||||
Q 40.234375 -1.421875 26.90625 -1.421875
|
||||
Q 22.3125 -1.421875 17.546875 -0.640625
|
||||
Q 12.796875 0.140625 7.71875 1.703125
|
||||
L 7.71875 11.625
|
||||
Q 12.109375 9.234375 16.796875 8.0625
|
||||
Q 21.484375 6.890625 26.703125 6.890625
|
||||
Q 35.15625 6.890625 40.078125 11.328125
|
||||
Q 45.015625 15.765625 45.015625 23.390625
|
||||
Q 45.015625 31 40.078125 35.4375
|
||||
Q 35.15625 39.890625 26.703125 39.890625
|
||||
Q 22.75 39.890625 18.8125 39.015625
|
||||
Q 14.890625 38.140625 10.796875 36.28125
|
||||
z
|
||||
" id="DejaVuSans-35"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(287.445277 270.918437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-35"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_7">
|
||||
<g id="line2d_14">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 336.555562 256.32
|
||||
L 336.555562 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_15">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="336.555562" xlink:href="#m894b1b7ca2" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_7">
|
||||
<!-- 6 -->
|
||||
<defs>
|
||||
<path d="M 33.015625 40.375
|
||||
Q 26.375 40.375 22.484375 35.828125
|
||||
Q 18.609375 31.296875 18.609375 23.390625
|
||||
Q 18.609375 15.53125 22.484375 10.953125
|
||||
Q 26.375 6.390625 33.015625 6.390625
|
||||
Q 39.65625 6.390625 43.53125 10.953125
|
||||
Q 47.40625 15.53125 47.40625 23.390625
|
||||
Q 47.40625 31.296875 43.53125 35.828125
|
||||
Q 39.65625 40.375 33.015625 40.375
|
||||
M 52.59375 71.296875
|
||||
L 52.59375 62.3125
|
||||
Q 48.875 64.0625 45.09375 64.984375
|
||||
Q 41.3125 65.921875 37.59375 65.921875
|
||||
Q 27.828125 65.921875 22.671875 59.328125
|
||||
Q 17.53125 52.734375 16.796875 39.40625
|
||||
Q 19.671875 43.65625 24.015625 45.921875
|
||||
Q 28.375 48.1875 33.59375 48.1875
|
||||
Q 44.578125 48.1875 50.953125 41.515625
|
||||
Q 57.328125 34.859375 57.328125 23.390625
|
||||
Q 57.328125 12.15625 50.6875 5.359375
|
||||
Q 44.046875 -1.421875 33.015625 -1.421875
|
||||
Q 20.359375 -1.421875 13.671875 8.265625
|
||||
Q 6.984375 17.96875 6.984375 36.375
|
||||
Q 6.984375 53.65625 15.1875 63.9375
|
||||
Q 23.390625 74.21875 37.203125 74.21875
|
||||
Q 40.921875 74.21875 44.703125 73.484375
|
||||
Q 48.484375 72.75 52.59375 71.296875
|
||||
" id="DejaVuSans-36"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(333.374312 270.918437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-36"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_8">
|
||||
<g id="line2d_16">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 382.484596 256.32
|
||||
L 382.484596 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_17">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="382.484596" xlink:href="#m894b1b7ca2" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_8">
|
||||
<!-- 7 -->
|
||||
<defs>
|
||||
<path d="M 8.203125 72.90625
|
||||
L 55.078125 72.90625
|
||||
L 55.078125 68.703125
|
||||
L 28.609375 0
|
||||
L 18.3125 0
|
||||
L 43.21875 64.59375
|
||||
L 8.203125 64.59375
|
||||
z
|
||||
" id="DejaVuSans-37"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(379.303346 270.918437)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-37"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_9">
|
||||
<g id="line2d_18">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L 0 2
|
||||
" id="m6888e25205" style="stroke:#01769d;stroke-width:0.6;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_10">
|
||||
<g id="line2d_19">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="106.91039" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_11">
|
||||
<g id="line2d_20">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="133.777152" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_12">
|
||||
<g id="line2d_21">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="152.839424" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_13">
|
||||
<g id="line2d_22">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="167.625271" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_14">
|
||||
<g id="line2d_23">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="179.706187" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_15">
|
||||
<g id="line2d_24">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="189.920456" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_16">
|
||||
<g id="line2d_25">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="198.768458" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_17">
|
||||
<g id="line2d_26">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="206.57295" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_18">
|
||||
<g id="line2d_27">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="259.483339" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_19">
|
||||
<g id="line2d_28">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="286.350102" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_20">
|
||||
<g id="line2d_29">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="305.412374" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_21">
|
||||
<g id="line2d_30">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="320.19822" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_22">
|
||||
<g id="line2d_31">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="332.279137" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_23">
|
||||
<g id="line2d_32">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="342.493406" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_24">
|
||||
<g id="line2d_33">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="351.341408" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_25">
|
||||
<g id="line2d_34">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="359.145899" xlink:href="#m6888e25205" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_9">
|
||||
<!-- bit index -->
|
||||
<defs>
|
||||
<path d="M 48.6875 27.296875
|
||||
Q 48.6875 37.203125 44.609375 42.84375
|
||||
Q 40.53125 48.484375 33.40625 48.484375
|
||||
Q 26.265625 48.484375 22.1875 42.84375
|
||||
Q 18.109375 37.203125 18.109375 27.296875
|
||||
Q 18.109375 17.390625 22.1875 11.75
|
||||
Q 26.265625 6.109375 33.40625 6.109375
|
||||
Q 40.53125 6.109375 44.609375 11.75
|
||||
Q 48.6875 17.390625 48.6875 27.296875
|
||||
M 18.109375 46.390625
|
||||
Q 20.953125 51.265625 25.265625 53.625
|
||||
Q 29.59375 56 35.59375 56
|
||||
Q 45.5625 56 51.78125 48.09375
|
||||
Q 58.015625 40.1875 58.015625 27.296875
|
||||
Q 58.015625 14.40625 51.78125 6.484375
|
||||
Q 45.5625 -1.421875 35.59375 -1.421875
|
||||
Q 29.59375 -1.421875 25.265625 0.953125
|
||||
Q 20.953125 3.328125 18.109375 8.203125
|
||||
L 18.109375 0
|
||||
L 9.078125 0
|
||||
L 9.078125 75.984375
|
||||
L 18.109375 75.984375
|
||||
z
|
||||
" id="DejaVuSans-62"/>
|
||||
<path d="M 9.421875 54.6875
|
||||
L 18.40625 54.6875
|
||||
L 18.40625 0
|
||||
L 9.421875 0
|
||||
z
|
||||
M 9.421875 75.984375
|
||||
L 18.40625 75.984375
|
||||
L 18.40625 64.59375
|
||||
L 9.421875 64.59375
|
||||
z
|
||||
" id="DejaVuSans-69"/>
|
||||
<path d="M 18.3125 70.21875
|
||||
L 18.3125 54.6875
|
||||
L 36.8125 54.6875
|
||||
L 36.8125 47.703125
|
||||
L 18.3125 47.703125
|
||||
L 18.3125 18.015625
|
||||
Q 18.3125 11.328125 20.140625 9.421875
|
||||
Q 21.96875 7.515625 27.59375 7.515625
|
||||
L 36.8125 7.515625
|
||||
L 36.8125 0
|
||||
L 27.59375 0
|
||||
Q 17.1875 0 13.234375 3.875
|
||||
Q 9.28125 7.765625 9.28125 18.015625
|
||||
L 9.28125 47.703125
|
||||
L 2.6875 47.703125
|
||||
L 2.6875 54.6875
|
||||
L 9.28125 54.6875
|
||||
L 9.28125 70.21875
|
||||
z
|
||||
" id="DejaVuSans-74"/>
|
||||
<path id="DejaVuSans-20"/>
|
||||
<path d="M 54.890625 33.015625
|
||||
L 54.890625 0
|
||||
L 45.90625 0
|
||||
L 45.90625 32.71875
|
||||
Q 45.90625 40.484375 42.875 44.328125
|
||||
Q 39.84375 48.1875 33.796875 48.1875
|
||||
Q 26.515625 48.1875 22.3125 43.546875
|
||||
Q 18.109375 38.921875 18.109375 30.90625
|
||||
L 18.109375 0
|
||||
L 9.078125 0
|
||||
L 9.078125 54.6875
|
||||
L 18.109375 54.6875
|
||||
L 18.109375 46.1875
|
||||
Q 21.34375 51.125 25.703125 53.5625
|
||||
Q 30.078125 56 35.796875 56
|
||||
Q 45.21875 56 50.046875 50.171875
|
||||
Q 54.890625 44.34375 54.890625 33.015625
|
||||
" id="DejaVuSans-6e"/>
|
||||
<path d="M 45.40625 46.390625
|
||||
L 45.40625 75.984375
|
||||
L 54.390625 75.984375
|
||||
L 54.390625 0
|
||||
L 45.40625 0
|
||||
L 45.40625 8.203125
|
||||
Q 42.578125 3.328125 38.25 0.953125
|
||||
Q 33.9375 -1.421875 27.875 -1.421875
|
||||
Q 17.96875 -1.421875 11.734375 6.484375
|
||||
Q 5.515625 14.40625 5.515625 27.296875
|
||||
Q 5.515625 40.1875 11.734375 48.09375
|
||||
Q 17.96875 56 27.875 56
|
||||
Q 33.9375 56 38.25 53.625
|
||||
Q 42.578125 51.265625 45.40625 46.390625
|
||||
M 14.796875 27.296875
|
||||
Q 14.796875 17.390625 18.875 11.75
|
||||
Q 22.953125 6.109375 30.078125 6.109375
|
||||
Q 37.203125 6.109375 41.296875 11.75
|
||||
Q 45.40625 17.390625 45.40625 27.296875
|
||||
Q 45.40625 37.203125 41.296875 42.84375
|
||||
Q 37.203125 48.484375 30.078125 48.484375
|
||||
Q 22.953125 48.484375 18.875 42.84375
|
||||
Q 14.796875 37.203125 14.796875 27.296875
|
||||
" id="DejaVuSans-64"/>
|
||||
<path d="M 56.203125 29.59375
|
||||
L 56.203125 25.203125
|
||||
L 14.890625 25.203125
|
||||
Q 15.484375 15.921875 20.484375 11.0625
|
||||
Q 25.484375 6.203125 34.421875 6.203125
|
||||
Q 39.59375 6.203125 44.453125 7.46875
|
||||
Q 49.3125 8.734375 54.109375 11.28125
|
||||
L 54.109375 2.78125
|
||||
Q 49.265625 0.734375 44.1875 -0.34375
|
||||
Q 39.109375 -1.421875 33.890625 -1.421875
|
||||
Q 20.796875 -1.421875 13.15625 6.1875
|
||||
Q 5.515625 13.8125 5.515625 26.8125
|
||||
Q 5.515625 40.234375 12.765625 48.109375
|
||||
Q 20.015625 56 32.328125 56
|
||||
Q 43.359375 56 49.78125 48.890625
|
||||
Q 56.203125 41.796875 56.203125 29.59375
|
||||
M 47.21875 32.234375
|
||||
Q 47.125 39.59375 43.09375 43.984375
|
||||
Q 39.0625 48.390625 32.421875 48.390625
|
||||
Q 24.90625 48.390625 20.390625 44.140625
|
||||
Q 15.875 39.890625 15.1875 32.171875
|
||||
z
|
||||
" id="DejaVuSans-65"/>
|
||||
<path d="M 54.890625 54.6875
|
||||
L 35.109375 28.078125
|
||||
L 55.90625 0
|
||||
L 45.3125 0
|
||||
L 29.390625 21.484375
|
||||
L 13.484375 0
|
||||
L 2.875 0
|
||||
L 24.125 28.609375
|
||||
L 4.6875 54.6875
|
||||
L 15.28125 54.6875
|
||||
L 29.78125 35.203125
|
||||
L 44.28125 54.6875
|
||||
z
|
||||
" id="DejaVuSans-78"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(199.520312 284.596563)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-62"/>
|
||||
<use x="63.476562" xlink:href="#DejaVuSans-69"/>
|
||||
<use x="91.259766" xlink:href="#DejaVuSans-74"/>
|
||||
<use x="130.46875" xlink:href="#DejaVuSans-20"/>
|
||||
<use x="162.255859" xlink:href="#DejaVuSans-69"/>
|
||||
<use x="190.039062" xlink:href="#DejaVuSans-6e"/>
|
||||
<use x="253.417969" xlink:href="#DejaVuSans-64"/>
|
||||
<use x="316.894531" xlink:href="#DejaVuSans-65"/>
|
||||
<use x="378.402344" xlink:href="#DejaVuSans-78"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="matplotlib.axis_2">
|
||||
<g id="ytick_1">
|
||||
<g id="line2d_35">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 54 256.32
|
||||
L 388.8 256.32
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_36">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L -3.5 0
|
||||
" id="m57c95a4823" style="stroke:#01769d;stroke-width:0.8;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m57c95a4823" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_10">
|
||||
<!-- 1 -->
|
||||
<g style="fill:#01769d;" transform="translate(40.6375 260.119219)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-31"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_2">
|
||||
<g id="line2d_37">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 54 214.820157
|
||||
L 388.8 214.820157
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_38">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m57c95a4823" y="214.820157"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_11">
|
||||
<!-- 2 -->
|
||||
<g style="fill:#01769d;" transform="translate(40.6375 218.619375)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_3">
|
||||
<g id="line2d_39">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 54 173.320313
|
||||
L 388.8 173.320313
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_40">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m57c95a4823" y="173.320313"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_12">
|
||||
<!-- 4 -->
|
||||
<g style="fill:#01769d;" transform="translate(40.6375 177.119532)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-34"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_4">
|
||||
<g id="line2d_41">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 54 131.82047
|
||||
L 388.8 131.82047
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_42">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m57c95a4823" y="131.82047"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_13">
|
||||
<!-- 8 -->
|
||||
<defs>
|
||||
<path d="M 31.78125 34.625
|
||||
Q 24.75 34.625 20.71875 30.859375
|
||||
Q 16.703125 27.09375 16.703125 20.515625
|
||||
Q 16.703125 13.921875 20.71875 10.15625
|
||||
Q 24.75 6.390625 31.78125 6.390625
|
||||
Q 38.8125 6.390625 42.859375 10.171875
|
||||
Q 46.921875 13.96875 46.921875 20.515625
|
||||
Q 46.921875 27.09375 42.890625 30.859375
|
||||
Q 38.875 34.625 31.78125 34.625
|
||||
M 21.921875 38.8125
|
||||
Q 15.578125 40.375 12.03125 44.71875
|
||||
Q 8.5 49.078125 8.5 55.328125
|
||||
Q 8.5 64.0625 14.71875 69.140625
|
||||
Q 20.953125 74.21875 31.78125 74.21875
|
||||
Q 42.671875 74.21875 48.875 69.140625
|
||||
Q 55.078125 64.0625 55.078125 55.328125
|
||||
Q 55.078125 49.078125 51.53125 44.71875
|
||||
Q 48 40.375 41.703125 38.8125
|
||||
Q 48.828125 37.15625 52.796875 32.3125
|
||||
Q 56.78125 27.484375 56.78125 20.515625
|
||||
Q 56.78125 9.90625 50.3125 4.234375
|
||||
Q 43.84375 -1.421875 31.78125 -1.421875
|
||||
Q 19.734375 -1.421875 13.25 4.234375
|
||||
Q 6.78125 9.90625 6.78125 20.515625
|
||||
Q 6.78125 27.484375 10.78125 32.3125
|
||||
Q 14.796875 37.15625 21.921875 38.8125
|
||||
M 18.3125 54.390625
|
||||
Q 18.3125 48.734375 21.84375 45.5625
|
||||
Q 25.390625 42.390625 31.78125 42.390625
|
||||
Q 38.140625 42.390625 41.71875 45.5625
|
||||
Q 45.3125 48.734375 45.3125 54.390625
|
||||
Q 45.3125 60.0625 41.71875 63.234375
|
||||
Q 38.140625 66.40625 31.78125 66.40625
|
||||
Q 25.390625 66.40625 21.84375 63.234375
|
||||
Q 18.3125 60.0625 18.3125 54.390625
|
||||
" id="DejaVuSans-38"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(40.6375 135.619688)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-38"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_5">
|
||||
<g id="line2d_43">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 54 90.320626
|
||||
L 388.8 90.320626
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_44">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m57c95a4823" y="90.320626"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_14">
|
||||
<!-- 16 -->
|
||||
<g style="fill:#01769d;" transform="translate(34.275 94.119845)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-31"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-36"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_6">
|
||||
<g id="line2d_45">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 54 48.820783
|
||||
L 388.8 48.820783
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_46">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m57c95a4823" y="48.820783"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_15">
|
||||
<!-- 32 -->
|
||||
<g style="fill:#01769d;" transform="translate(34.275 52.620002)scale(0.1 -0.1)">
|
||||
<use xlink:href="#DejaVuSans-33"/>
|
||||
<use x="63.623047" xlink:href="#DejaVuSans-32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_7">
|
||||
<g id="line2d_47">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L -2 0
|
||||
" id="m401704d485" style="stroke:#01769d;stroke-width:0.6;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="214.820157"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_8">
|
||||
<g id="line2d_48">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="190.544304"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_9">
|
||||
<g id="line2d_49">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="173.320313"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_10">
|
||||
<g id="line2d_50">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="159.960348"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_11">
|
||||
<g id="line2d_51">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="149.044461"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_12">
|
||||
<g id="line2d_52">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="139.81521"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_13">
|
||||
<g id="line2d_53">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="131.82047"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_14">
|
||||
<g id="line2d_54">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="124.768609"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_15">
|
||||
<g id="line2d_55">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="76.960661"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_16">
|
||||
<g id="line2d_56">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="52.684809"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_17">
|
||||
<g id="line2d_57">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m401704d485" y="35.460817"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="line2d_58">
|
||||
<path clip-path="url(#pb2ddf07cb0)" d="M 60.981355 253.94359
|
||||
L 106.91039 239.555517
|
||||
L 152.839424 218.25662
|
||||
L 198.768458 190.165617
|
||||
L 244.697493 156.695449
|
||||
L 290.626527 119.654078
|
||||
L 336.555562 80.514491
|
||||
L 382.484596 40.23061
|
||||
" style="fill:none;stroke:#fe3ea0;stroke-linecap:square;stroke-width:1.5;"/>
|
||||
</g>
|
||||
<g id="patch_3">
|
||||
<path d="M 54 256.32
|
||||
L 54 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-linecap:square;stroke-linejoin:miter;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="patch_4">
|
||||
<path d="M 54 256.32
|
||||
L 388.8 256.32
|
||||
" style="fill:none;stroke:#08bdf9;stroke-linecap:square;stroke-linejoin:miter;stroke-width:0.8;"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<defs>
|
||||
<clipPath id="pb2ddf07cb0">
|
||||
<rect height="221.76" width="334.8" x="54" y="34.56"/>
|
||||
</clipPath>
|
||||
</defs>
|
||||
</svg>
|
||||
|
Before Width: | Height: | Size: 29 KiB |
|
Before Width: | Height: | Size: 856 KiB |
|
Before Width: | Height: | Size: 1.2 MiB |
|
Before Width: | Height: | Size: 279 KiB |
|
Before Width: | Height: | Size: 285 KiB |
|
Before Width: | Height: | Size: 1.7 MiB |
|
Before Width: | Height: | Size: 1.9 MiB |
|
Before Width: | Height: | Size: 1.3 MiB |
|
Before Width: | Height: | Size: 134 KiB |
|
Before Width: | Height: | Size: 354 KiB |
|
Before Width: | Height: | Size: 58 KiB |
|
Before Width: | Height: | Size: 56 KiB |
|
Before Width: | Height: | Size: 820 KiB |
|
|
@ -1,727 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8" standalone="no"?>
|
||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN"
|
||||
"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||
<!-- Created with matplotlib (http://matplotlib.org/) -->
|
||||
<svg height="288pt" version="1.1" viewBox="0 0 432 288" width="432pt" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||
<defs>
|
||||
<style type="text/css">
|
||||
*{stroke-linecap:butt;stroke-linejoin:round;}
|
||||
</style>
|
||||
</defs>
|
||||
<g id="figure_1">
|
||||
<g id="patch_1">
|
||||
<path d="M 0 288
|
||||
L 432 288
|
||||
L 432 0
|
||||
L 0 0
|
||||
z
|
||||
" style="fill:none;opacity:0;"/>
|
||||
</g>
|
||||
<g id="axes_1">
|
||||
<g id="patch_2">
|
||||
<path d="M 54 256.32
|
||||
L 388.8 256.32
|
||||
L 388.8 34.56
|
||||
L 54 34.56
|
||||
z
|
||||
" style="fill:#ffffff;"/>
|
||||
</g>
|
||||
<g id="matplotlib.axis_1">
|
||||
<g id="xtick_1">
|
||||
<g id="line2d_1">
|
||||
<path clip-path="url(#p2822336f64)" d="M 54 256.32
|
||||
L 54 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_2">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L 0 3.5
|
||||
" id="m97f003056d" style="stroke:#01769d;stroke-width:0.8;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#m97f003056d" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_1">
|
||||
<!-- ${10^{0}}$ -->
|
||||
<defs>
|
||||
<path d="M 12.40625 8.296875
|
||||
L 28.515625 8.296875
|
||||
L 28.515625 63.921875
|
||||
L 10.984375 60.40625
|
||||
L 10.984375 69.390625
|
||||
L 28.421875 72.90625
|
||||
L 38.28125 72.90625
|
||||
L 38.28125 8.296875
|
||||
L 54.390625 8.296875
|
||||
L 54.390625 0
|
||||
L 12.40625 0
|
||||
z
|
||||
" id="DejaVuSans-31"/>
|
||||
<path d="M 31.78125 66.40625
|
||||
Q 24.171875 66.40625 20.328125 58.90625
|
||||
Q 16.5 51.421875 16.5 36.375
|
||||
Q 16.5 21.390625 20.328125 13.890625
|
||||
Q 24.171875 6.390625 31.78125 6.390625
|
||||
Q 39.453125 6.390625 43.28125 13.890625
|
||||
Q 47.125 21.390625 47.125 36.375
|
||||
Q 47.125 51.421875 43.28125 58.90625
|
||||
Q 39.453125 66.40625 31.78125 66.40625
|
||||
M 31.78125 74.21875
|
||||
Q 44.046875 74.21875 50.515625 64.515625
|
||||
Q 56.984375 54.828125 56.984375 36.375
|
||||
Q 56.984375 17.96875 50.515625 8.265625
|
||||
Q 44.046875 -1.421875 31.78125 -1.421875
|
||||
Q 19.53125 -1.421875 13.0625 8.265625
|
||||
Q 6.59375 17.96875 6.59375 36.375
|
||||
Q 6.59375 54.828125 13.0625 64.515625
|
||||
Q 19.53125 74.21875 31.78125 74.21875
|
||||
" id="DejaVuSans-30"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(45.2 270.918437)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.765625)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.765625)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 39.046875)scale(0.7)" xlink:href="#DejaVuSans-30"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_2">
|
||||
<g id="line2d_3">
|
||||
<path clip-path="url(#p2822336f64)" d="M 193.022691 256.32
|
||||
L 193.022691 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_4">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="193.022691" xlink:href="#m97f003056d" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_2">
|
||||
<!-- ${10^{1}}$ -->
|
||||
<g style="fill:#01769d;" transform="translate(184.222691 270.918437)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.684375)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.684375)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 38.965625)scale(0.7)" xlink:href="#DejaVuSans-31"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_3">
|
||||
<g id="line2d_5">
|
||||
<path clip-path="url(#p2822336f64)" d="M 332.045382 256.32
|
||||
L 332.045382 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_6">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="332.045382" xlink:href="#m97f003056d" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_3">
|
||||
<!-- ${10^{2}}$ -->
|
||||
<defs>
|
||||
<path d="M 19.1875 8.296875
|
||||
L 53.609375 8.296875
|
||||
L 53.609375 0
|
||||
L 7.328125 0
|
||||
L 7.328125 8.296875
|
||||
Q 12.9375 14.109375 22.625 23.890625
|
||||
Q 32.328125 33.6875 34.8125 36.53125
|
||||
Q 39.546875 41.84375 41.421875 45.53125
|
||||
Q 43.3125 49.21875 43.3125 52.78125
|
||||
Q 43.3125 58.59375 39.234375 62.25
|
||||
Q 35.15625 65.921875 28.609375 65.921875
|
||||
Q 23.96875 65.921875 18.8125 64.3125
|
||||
Q 13.671875 62.703125 7.8125 59.421875
|
||||
L 7.8125 69.390625
|
||||
Q 13.765625 71.78125 18.9375 73
|
||||
Q 24.125 74.21875 28.421875 74.21875
|
||||
Q 39.75 74.21875 46.484375 68.546875
|
||||
Q 53.21875 62.890625 53.21875 53.421875
|
||||
Q 53.21875 48.921875 51.53125 44.890625
|
||||
Q 49.859375 40.875 45.40625 35.40625
|
||||
Q 44.1875 33.984375 37.640625 27.21875
|
||||
Q 31.109375 20.453125 19.1875 8.296875
|
||||
" id="DejaVuSans-32"/>
|
||||
</defs>
|
||||
<g style="fill:#01769d;" transform="translate(323.245382 270.918437)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.765625)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.765625)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 39.046875)scale(0.7)" xlink:href="#DejaVuSans-32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_4">
|
||||
<g id="line2d_7">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L 0 2
|
||||
" id="m382e56d781" style="stroke:#01769d;stroke-width:0.6;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="95.85" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_5">
|
||||
<g id="line2d_8">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="120.330681" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_6">
|
||||
<g id="line2d_9">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="137.7" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_7">
|
||||
<g id="line2d_10">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="151.172691" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_8">
|
||||
<g id="line2d_11">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="162.180681" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_9">
|
||||
<g id="line2d_12">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="171.487803" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_10">
|
||||
<g id="line2d_13">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="179.55" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_11">
|
||||
<g id="line2d_14">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="186.661361" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_12">
|
||||
<g id="line2d_15">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="234.872691" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_13">
|
||||
<g id="line2d_16">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="259.353371" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_14">
|
||||
<g id="line2d_17">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="276.722691" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_15">
|
||||
<g id="line2d_18">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="290.195382" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_16">
|
||||
<g id="line2d_19">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="301.203371" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_17">
|
||||
<g id="line2d_20">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="310.510494" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_18">
|
||||
<g id="line2d_21">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="318.572691" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_19">
|
||||
<g id="line2d_22">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="325.684052" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="xtick_20">
|
||||
<g id="line2d_23">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="373.895382" xlink:href="#m382e56d781" y="256.32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="matplotlib.axis_2">
|
||||
<g id="ytick_1">
|
||||
<g id="line2d_24">
|
||||
<path clip-path="url(#p2822336f64)" d="M 54 197.74235
|
||||
L 388.8 197.74235
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_25">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L -3.5 0
|
||||
" id="mb373a1162c" style="stroke:#01769d;stroke-width:0.8;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#mb373a1162c" y="197.74235"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_4">
|
||||
<!-- ${10^{1}}$ -->
|
||||
<g style="fill:#01769d;" transform="translate(29.4 201.541568)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.684375)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.684375)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 38.965625)scale(0.7)" xlink:href="#DejaVuSans-31"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_2">
|
||||
<g id="line2d_26">
|
||||
<path clip-path="url(#p2822336f64)" d="M 54 91.371981
|
||||
L 388.8 91.371981
|
||||
" style="fill:none;stroke:#08bdf9;stroke-dasharray:2.2,2.2;stroke-dashoffset:0;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="line2d_27">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.8;" x="54" xlink:href="#mb373a1162c" y="91.371981"/>
|
||||
</g>
|
||||
</g>
|
||||
<g id="text_5">
|
||||
<!-- ${10^{2}}$ -->
|
||||
<g style="fill:#01769d;" transform="translate(29.4 95.171199)scale(0.1 -0.1)">
|
||||
<use transform="translate(0 0.765625)" xlink:href="#DejaVuSans-31"/>
|
||||
<use transform="translate(63.623047 0.765625)" xlink:href="#DejaVuSans-30"/>
|
||||
<use transform="translate(128.203125 39.046875)scale(0.7)" xlink:href="#DejaVuSans-32"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_3">
|
||||
<g id="line2d_28">
|
||||
<defs>
|
||||
<path d="M 0 0
|
||||
L -2 0
|
||||
" id="m09a18ae090" style="stroke:#01769d;stroke-width:0.6;"/>
|
||||
</defs>
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="253.361155"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_4">
|
||||
<g id="line2d_29">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="240.071375"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_5">
|
||||
<g id="line2d_30">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="229.763021"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_6">
|
||||
<g id="line2d_31">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="221.340483"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_7">
|
||||
<g id="line2d_32">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="214.219328"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_8">
|
||||
<g id="line2d_33">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="208.050704"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_9">
|
||||
<g id="line2d_34">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="202.609591"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_10">
|
||||
<g id="line2d_35">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="165.721678"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_11">
|
||||
<g id="line2d_36">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="146.990786"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_12">
|
||||
<g id="line2d_37">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="133.701006"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_13">
|
||||
<g id="line2d_38">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="123.392652"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_14">
|
||||
<g id="line2d_39">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="114.970114"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_15">
|
||||
<g id="line2d_40">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="107.848959"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_16">
|
||||
<g id="line2d_41">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="101.680335"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_17">
|
||||
<g id="line2d_42">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="96.239222"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_18">
|
||||
<g id="line2d_43">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="59.351309"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="ytick_19">
|
||||
<g id="line2d_44">
|
||||
<g>
|
||||
<use style="fill:#01769d;stroke:#01769d;stroke-width:0.6;" x="54" xlink:href="#m09a18ae090" y="40.620417"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<g id="line2d_45">
|
||||
<path clip-path="url(#p2822336f64)" d="M -1 288.398444
|
||||
L 54 246.24
|
||||
L 95.85 234.630263
|
||||
L 120.330681 208.050704
|
||||
L 137.7 217.642825
|
||||
L 151.172691 197.74235
|
||||
L 162.180681 193.339394
|
||||
L 171.487803 180.577576
|
||||
L 179.55 195.488436
|
||||
L 186.661361 182.198657
|
||||
L 193.022691 179.011457
|
||||
L 198.777213 169.323194
|
||||
L 204.030681 173.229413
|
||||
L 208.863402 164.580976
|
||||
L 213.337803 162.380748
|
||||
L 217.503371 155.413324
|
||||
L 221.4 169.323194
|
||||
L 225.06032 161.318723
|
||||
L 228.511361 159.265226
|
||||
L 231.775766 152.72153
|
||||
L 234.872691 155.413324
|
||||
L 237.818484 149.360334
|
||||
L 240.627213 147.767208
|
||||
L 243.311068 142.587831
|
||||
L 245.880681 148.556904
|
||||
L 248.345382 143.293128
|
||||
L 250.713402 141.89314
|
||||
L 252.992042 137.302523
|
||||
L 255.187803 139.214357
|
||||
L 257.306505 134.870589
|
||||
L 259.353371 133.701006
|
||||
L 261.333116 129.826012
|
||||
L 263.25 140.534334
|
||||
L 265.107894 136.070555
|
||||
L 266.91032 134.870589
|
||||
L 268.660494 130.900387
|
||||
L 270.361361 132.560305
|
||||
L 272.015623 128.776056
|
||||
L 273.625766 127.749435
|
||||
L 275.194083 124.325938
|
||||
L 276.722691 128.259894
|
||||
L 278.213551 124.799748
|
||||
L 279.668484 123.856938
|
||||
L 281.08918 120.700858
|
||||
L 282.477213 122.027152
|
||||
L 283.834052 118.989697
|
||||
L 285.161068 118.157313
|
||||
L 286.459543 115.356694
|
||||
L 287.730681 120.700858
|
||||
L 288.975607 117.746679
|
||||
L 290.195382 116.936201
|
||||
L 291.391001 114.206525
|
||||
L 292.563402 115.356694
|
||||
L 293.713471 112.716201
|
||||
L 294.842042 111.988688
|
||||
L 295.949904 109.529001
|
||||
L 297.037803 112.351012
|
||||
L 298.106447 109.872468
|
||||
L 299.156505 109.188069
|
||||
L 300.188612 106.869501
|
||||
L 301.203371 107.848959
|
||||
L 302.201358 105.595046
|
||||
L 303.183116 104.970765
|
||||
L 304.149165 102.849918
|
||||
L 305.1 110.218508
|
||||
L 306.036093 107.848959
|
||||
L 306.957894 107.193685
|
||||
L 307.865833 104.970765
|
||||
L 308.76032 105.910379
|
||||
L 309.641749 103.746957
|
||||
L 310.510494 103.146999
|
||||
L 311.366917 101.106463
|
||||
L 312.211361 103.446004
|
||||
L 313.044158 101.392508
|
||||
L 313.865623 100.822179
|
||||
L 314.676062 98.879716
|
||||
L 315.475766 99.702219
|
||||
L 316.265017 97.80534
|
||||
L 317.044083 97.277379
|
||||
L 317.813224 95.475633
|
||||
L 318.572691 98.879716
|
||||
L 319.322723 97.015645
|
||||
L 320.063551 96.496582
|
||||
L 320.7954 94.724461
|
||||
L 321.518484 95.475633
|
||||
L 322.233011 93.741529
|
||||
L 322.93918 93.257796
|
||||
L 323.637185 91.60354
|
||||
L 324.327213 93.499029
|
||||
L 325.009444 91.836267
|
||||
L 325.684052 91.371981
|
||||
L 326.351206 89.78277
|
||||
L 327.011068 90.457177
|
||||
L 327.663796 88.898608
|
||||
L 328.309543 88.462793
|
||||
L 328.948457 86.969026
|
||||
L 329.580681 90.912315
|
||||
L 330.206352 89.338574
|
||||
L 330.825607 88.898608
|
||||
L 331.438575 87.39091
|
||||
L 332.045382 88.031051
|
||||
L 332.646151 86.550959
|
||||
L 333.241001 86.136641
|
||||
L 333.830047 84.715112
|
||||
L 334.413402 86.343336
|
||||
L 334.991175 84.915529
|
||||
L 335.563471 84.515561
|
||||
L 336.130393 83.142328
|
||||
L 336.692042 83.725865
|
||||
L 337.248514 82.375571
|
||||
L 337.799904 81.996912
|
||||
L 338.346304 80.695529
|
||||
L 338.887803 83.142328
|
||||
L 339.42449 81.808741
|
||||
L 339.956447 81.434682
|
||||
L 340.483759 80.148823
|
||||
L 341.006505 80.695529
|
||||
L 341.524764 79.429804
|
||||
L 342.038612 79.074448
|
||||
L 342.548123 77.851796
|
||||
L 343.053371 79.251784
|
||||
L 343.554426 78.024492
|
||||
L 344.051358 77.679744
|
||||
L 344.544232 76.492991
|
||||
L 345.033116 76.997871
|
||||
L 345.518072 75.828288
|
||||
L 345.999165 75.499489
|
||||
L 346.476454 74.366779
|
||||
L 346.95 79.074448
|
||||
L 347.419861 77.851796
|
||||
L 347.886093 77.50833
|
||||
L 348.348753 76.325916
|
||||
L 348.807894 76.828964
|
||||
L 349.26357 75.663596
|
||||
L 349.715833 75.335963
|
||||
L 350.164733 74.207207
|
||||
L 350.61032 75.499489
|
||||
L 351.052643 74.366779
|
||||
L 351.491749 74.048184
|
||||
L 351.927684 72.950093
|
||||
L 352.360494 73.417511
|
||||
L 352.790224 72.334137
|
||||
L 353.216917 72.029211
|
||||
L 353.640615 70.977548
|
||||
L 354.061361 72.950093
|
||||
L 354.479195 71.8775
|
||||
L 354.894158 71.575564
|
||||
L 355.306288 70.534062
|
||||
L 355.715623 70.977548
|
||||
L 356.122203 69.949294
|
||||
L 356.526062 69.659663
|
||||
L 356.927238 68.660018
|
||||
L 357.325766 69.804252
|
||||
L 357.721681 68.801507
|
||||
L 358.115017 68.518961
|
||||
L 358.505806 67.543443
|
||||
L 358.894083 67.959002
|
||||
L 359.279878 66.995115
|
||||
L 359.663224 66.723373
|
||||
L 360.044152 65.784668
|
||||
L 360.422691 68.238133
|
||||
L 360.798871 67.268465
|
||||
L 361.172723 66.995115
|
||||
L 361.544273 66.050929
|
||||
L 361.913551 66.453219
|
||||
L 362.280585 65.519934
|
||||
L 362.6454 65.256708
|
||||
L 363.008025 64.347051
|
||||
L 363.368484 65.388133
|
||||
L 363.726804 64.475911
|
||||
L 364.083011 64.21855
|
||||
L 364.437128 63.328915
|
||||
L 364.78918 63.708091
|
||||
L 365.139191 62.82814
|
||||
L 365.487185 62.579773
|
||||
L 365.833185 61.720857
|
||||
L 366.177213 63.328915
|
||||
L 366.519292 62.456089
|
||||
L 366.859444 62.209709
|
||||
L 367.19769 61.357584
|
||||
L 367.534052 61.720857
|
||||
L 367.86855 60.87762
|
||||
L 368.201206 60.639496
|
||||
L 368.532038 59.815595
|
||||
L 368.861068 60.758405
|
||||
L 369.188314 59.932399
|
||||
L 369.513796 59.699085
|
||||
L 369.837533 58.891643
|
||||
L 370.159543 59.235963
|
||||
L 370.479845 58.436506
|
||||
L 370.798457 58.210608
|
||||
L 371.115396 57.428551
|
||||
L 371.430681 60.0495
|
||||
L 371.744327 59.235963
|
||||
L 372.056352 59.006131
|
||||
L 372.366774 58.210608
|
||||
L 372.675607 58.54987
|
||||
L 372.982869 57.762099
|
||||
L 373.288575 57.539466
|
||||
L 373.59274 56.768596
|
||||
L 373.895382 57.650648
|
||||
L 374.196513 56.877936
|
||||
L 374.496151 56.659515
|
||||
L 374.794308 55.903071
|
||||
L 375.091001 56.225746
|
||||
L 375.386242 55.476315
|
||||
L 375.680047 55.264406
|
||||
L 375.972429 54.530287
|
||||
L 376.263402 55.903071
|
||||
L 376.55298 55.158815
|
||||
L 376.841175 54.948354
|
||||
L 377.128001 54.219201
|
||||
L 377.413471 54.530287
|
||||
L 377.697598 53.807652
|
||||
L 377.980393 53.603245
|
||||
L 378.261871 52.894857
|
||||
L 378.542042 53.705335
|
||||
L 378.820919 52.995393
|
||||
L 379.098514 52.79454
|
||||
L 379.374838 52.098354
|
||||
L 379.649904 52.395436
|
||||
L 379.923722 51.705193
|
||||
L 380.196304 51.509861
|
||||
L 380.467661 50.832629
|
||||
L 380.737803 52.594557
|
||||
L 381.006743 51.901355
|
||||
L 381.27449 51.705193
|
||||
L 381.541054 51.025113
|
||||
L 381.806447 51.315351
|
||||
L 382.070679 50.640944
|
||||
L 382.333759 50.450051
|
||||
L 382.595697 49.788069
|
||||
L 382.856505 50.545399
|
||||
L 383.11619 49.882059
|
||||
L 383.374764 49.69427
|
||||
L 383.632234 49.042955
|
||||
L 383.888612 49.320966
|
||||
L 384.143905 48.674857
|
||||
L 384.398123 48.491902
|
||||
L 384.651276 47.857206
|
||||
L 384.903371 49.042955
|
||||
L 385.154419 48.400696
|
||||
L 385.404426 48.218821
|
||||
L 385.653403 47.587841
|
||||
L 385.901358 47.857206
|
||||
L 386.148298 47.231113
|
||||
L 386.394232 47.053776
|
||||
L 386.639169 46.438406
|
||||
L 386.883116 47.142359
|
||||
L 387.126081 46.525815
|
||||
L 387.368072 46.351161
|
||||
L 387.609098 45.745018
|
||||
L 387.849165 46.003821
|
||||
L 388.088281 45.402189
|
||||
L 388.326454 45.231723
|
||||
L 388.563691 44.64
|
||||
L 388.563691 44.64
|
||||
" style="fill:none;stroke:#fe3ea0;stroke-linecap:square;stroke-width:1.5;"/>
|
||||
</g>
|
||||
<g id="patch_3">
|
||||
<path d="M 54 256.32
|
||||
L 54 34.56
|
||||
" style="fill:none;stroke:#08bdf9;stroke-linecap:square;stroke-linejoin:miter;stroke-width:0.8;"/>
|
||||
</g>
|
||||
<g id="patch_4">
|
||||
<path d="M 54 256.32
|
||||
L 388.8 256.32
|
||||
" style="fill:none;stroke:#08bdf9;stroke-linecap:square;stroke-linejoin:miter;stroke-width:0.8;"/>
|
||||
</g>
|
||||
</g>
|
||||
</g>
|
||||
<defs>
|
||||
<clipPath id="p2822336f64">
|
||||
<rect height="221.76" width="334.8" x="54" y="34.56"/>
|
||||
</clipPath>
|
||||
</defs>
|
||||
</svg>
|
||||
|
Before Width: | Height: | Size: 20 KiB |
|
|
@ -1,456 +0,0 @@
|
|||
---
|
||||
title: "32-Channel LED tape driver"
|
||||
date: 2018-05-02T11:31:14+02:00
|
||||
---
|
||||
|
||||
Theoretical basics
|
||||
==================
|
||||
|
||||
Together, a friend and I outfitted the small staircase at Berlin's Chaos Computer Club with nice, shiny RGB-WW LED tape
|
||||
for ambient lighting. This tape is like regular RGB tape but with an additional warm white channel, which makes for much
|
||||
more natural pastels and whites. There are several variants of RGBW tape. Cheap ones have separate RGB and white LEDs,
|
||||
which is fine for indirect lighting but does not work for direct lighting. Since we wanted to mount our tape in channels
|
||||
at the front of the steps, we had to use the slightly more expensive variant with integrated RGBW LEDs. These are LEDs
|
||||
in the 5050 (5.0mm by 5.0mm) form factor common with RGB LEDs that have a small section divided off for the white
|
||||
channel. The red, green and blue LED chips sit together in the larger section covered with clear epoxy and the white
|
||||
channel is made up from the usual blue LED inside a yellow phosphor in the smaller section.
|
||||
|
||||
Since we wanted to light up all of 15 steps, and for greatest visual effect we would have liked to be able to control
|
||||
each step individually we had to find a way to control 60 channels of LED tape with a reasonable amount of hardware.
|
||||
|
||||
LED tape has integrated series resistors and runs off a fixed 12V or 24V constant-voltage supply. This means you don't
|
||||
need a complex constant-current driver as you'd need with high-power LEDs. You can just hook up a section of LED tape
|
||||
to a beefy MOSFET to control it. Traditionally, you would do *Pulse Width Modulation* (PWM) on the MOSFET's input to
|
||||
control the LED tape's brightness.
|
||||
|
||||
Pulse Width Modulation
|
||||
----------------------
|
||||
|
||||
`Pulse Width Modulation`_ is a technique of controlling the brightness of a load such as an LED with a digital signal.
|
||||
The basic idea is that if you turn the LED on and off much too fast for anyone to notice, you can control its power by
|
||||
changing how long you turn it on versus how long you leave it off.
|
||||
|
||||
PWM divides each second into a large number of periods. At the beginning of each period, you turn the LED on. After
|
||||
that, you wait a certain time until you turn it off. Then, you wait for the next period to begin. The periods are always
|
||||
the same length but you can set when you turn off the LED. If you turn it off right away, it's off almost all the time
|
||||
and it looks like it's off to your eye. If you turn it off right at the end, it's on almost all the time and it looks
|
||||
super bright to your eye. Now, if you turn it off halfway into the cycle, it's on half the time and it will look to your
|
||||
eye as half as bright as before. This means that you can control the LED's brightness with only a digital signal and
|
||||
good timing.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/pwm_schema.jpg" alt="A visualization of PWM at different duty cycles.">
|
||||
<figcaption>Waveforms of two PWM cycles at different duty cycles.</figcaption>
|
||||
</figure>
|
||||
|
||||
PWM works great if you have a dedicated PWM output on your microcontroller. It's extremely simple in both hardware and
|
||||
software. Unfortunately for us, controlling 32 channels with PWM is not that easy. Cheap microcontrollers only have `a
|
||||
handful of hardware PWM outputs`_, so we'd either have to do everything in software, bit-banging our LED modulation, or
|
||||
we'd have to use a dedicated chip.
|
||||
|
||||
Doing PWM in software is both error-prone and slow. Since the maximum dynamic range of a PWM signal is limited by the
|
||||
shortest duty cycle it can do, software PWM being slow means it has poor PWM resolution at maybe 8 bits at most. Poor
|
||||
color resolution is not a problem if all you're doing is to fade around the `HSV rainbow`_, but for ambient lighting
|
||||
where you *really* want to control the brightness down to a faint shimmer you need all the color resolution you can get.
|
||||
|
||||
If you rule out software PWM, what remains are dedicated `hardware PWM controllers`_. Most of these have either of three
|
||||
issues:
|
||||
|
||||
* They're expensive
|
||||
* They don't have generous PWM resolution either (12 bits if you're lucky)
|
||||
* They're meant to drive small LEDs such as a 7-segment display directly and you can't just hook up a MOSFET to their
|
||||
output
|
||||
|
||||
This means we're stuck in a dilemma between two poor solutions if we'd want to do PWM. Luckily for us, PWM is not the
|
||||
only modulation in town.
|
||||
|
||||
.. _`Pulse Width Modulation`: https://en.wikipedia.org/wiki/Pulse-width_modulation
|
||||
.. _`a handful of hardware PWM outputs`: https://www.nxp.com/parametricSearch#/&c=c731_c380_c173_c161_c163&page=1
|
||||
.. _`HSV rainbow`: https://en.wikipedia.org/wiki/HSL_and_HSV
|
||||
.. _`hardware PWM controllers`: http://www.ti.com/lit/ds/symlink/tlc5940.pdf
|
||||
|
||||
Binary Code Modulation
|
||||
----------------------
|
||||
|
||||
PWM is the bread-and-butter of the maker crowd. Everyone and their cat is doing it and it works really well most of the
|
||||
time. Unbeknownst to most of the maker crowd, there is however another popular modulation method that's mostly used in
|
||||
professional LED systems: Enter `*Binary Code Modulation* (BCM) <http://www.batsocks.co.uk/readme/art_bcm_1.htm>`_.
|
||||
|
||||
BCM is to PWM sort of what barcodes are to handwriting. While PWM is easy to understand and simple to implement if all
|
||||
you have is a counter and an IO pin, BCM is more complicated. On the other hand, computers can do complicated and BCM
|
||||
really shines in multi-channel applications.
|
||||
|
||||
Similar to PWM, BCM works by turning on and off the LED in short periods fast enough to make your eye perceive it as
|
||||
partially on all the time. In PWM the channel's brightness is linearly dependent on its duty cycle, i.e. the percentage
|
||||
it is turned on. In PWM the duty cycle D is the total period T divided by the on period T_on. The issue with doing PWM
|
||||
on many channels at once is that you have to turn off each channel at the exact time to match its duty cycle.
|
||||
Controlling many IO pins at once with precise timing is really hard to do in software.
|
||||
|
||||
BCM avoids this by further dividing each period into smaller periods which we'll call *bit periods* and splitting each
|
||||
channel's duty cycle into chunks the size of these bit periods. The amazingly elegant thing in BCM now is that as you
|
||||
can guess from the name these bit periods are weighted in powers of two. Say the shortest bit period lasts 1
|
||||
microsecond. Then the second-shortest bit period is 2 microseconds and the third is 4, the fifth 8, the sixth 16 and so
|
||||
on.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/bcm_schema.jpg" alt="A visualization of BCM at different duty cycles.">
|
||||
<figcaption>Waveforms of a single 4-bit BCM cycle at different duty cycles. This BCM can produce 16 different
|
||||
levels.</figcaption>
|
||||
</figure>
|
||||
|
||||
Staggered like this, you turn on the LED for integer value of microseconds by turning it on in the bit periods
|
||||
corresponding to the binary bits of that value. If I want my LED to light for 19 microseconds every period, I turn it on
|
||||
in the 16 microsecond bit period, the 2 microsecond bit period and the 1 microsecond bit period and leave it off for the
|
||||
4 and 8 mircosecond bit periods.
|
||||
|
||||
Now, how this is better instead of just more complicated than plain old PWM might not be clear yet. But consider this:
|
||||
Turning on and off a large number of channels, each at its own arbitrary time is hard because doing the timing in
|
||||
software is hard. We can't use hardware timers since we only have two or three of those, and we have 32 channels.
|
||||
However, we can use one hardware timer to trigger a really cheap external latch to turn on or off the 32 channels all at
|
||||
once. With this setup, we can only controll all channels at once, but we can do so with very precise timing.
|
||||
|
||||
All we need to do is to set our timer to the durations of the BCM bit periods, and we can get the same result as we'd
|
||||
get with PWM with only one hardware timer and a bit of code that is not timing-critical anymore.
|
||||
|
||||
Applications of Binary Code Modulation
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
BCM is a truly wondrous technique, and outside of hobbyist circles it is in fact very widely known. Though we're using
|
||||
it to control just 32 channels here, you can do much more channels without any problems. The most common application
|
||||
where BCM is invariably used is *any* kind of LED screen. Controlling the thousands and thousands of LEDs in an LED
|
||||
screen with PWM with a dedicated timer for each LED would not be feasible. With BCM, all you need to dedicate to a
|
||||
single LED is a flipflop (or part of one if you're multiplexing). In fact, there is a whole range of `ICs with no other
|
||||
purpose than to enable BCM on large LED matrices <http://www.vabolis.lt/stuff/MBI5026.pdf>`_. Basically, these are a
|
||||
high-speed shift register with latched outputs much like the venerable 74HC595_, only their outputs are constant-current
|
||||
sinks made so that you can directly connect an LED to them.
|
||||
|
||||
.. _74HC595: http://www.ti.com/lit/ds/symlink/sn74hc595.pdf
|
||||
|
||||
Running BCM on LED tape
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
In our case, we don't need any special driver chips to control our LED tape. We just connect the outputs of a 74HC595_
|
||||
shift register to one MOSFET_ each, and then we directly connect the LED tape to these MOSFETs. The MOSFETs allow us to
|
||||
drive a couple of amps into the LED tape from the weak outputs of the shift register.
|
||||
|
||||
The BCM timing is done by hooking up two timer channels of our microcontroller to the shift registers *strobe* and
|
||||
*reset* inputs. We set the timer to PWM mode so we can generate pulses with precise timing. At the beginning of each
|
||||
bit period, a pulse will strobe the data for this bit period that we shifted in previously. At the end of the bit
|
||||
period, one pulse will reset the shift register and one will strobe the freshly-reset zeros into the outputs.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/olsndot_output_schematic.jpg" alt="From left to right, we see the STM32, one of the shift
|
||||
registers, and the LEDs and MOSFETs. The LED tape is driven to ground by the MOSFETs, which are in turn directly
|
||||
driven from the shift register outputs. The shift register is wired up to the STM32 with its clock and data
|
||||
inputs on SCK and MOSI and its RESET and STROBE inputs on channel 2 and 3 of timer 1.">
|
||||
<figcaption>
|
||||
The schematic of a single output of this LED driver. Multiple shift register stages can be cascaded.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
|
||||
Our implementation of this system runs on an STM32F030F4P6_, the smallest, cheapest ARM microcontroller you can get from
|
||||
ST. This microcontroller has only 16kB of flash and 1kB of RAM, but that's plenty for our use. We use its SPI controller
|
||||
to feed the modulation data to the shift registers really fast, and we use two timer channels to control the shift
|
||||
registers' reset and strobe.
|
||||
|
||||
We can easily cascade shift registers without any ill side-effects, and even hundreds of channels should be no problem
|
||||
for this setup. The only reason we chose to stick to a 32-channel board is the mechanics of it. We thought it would be
|
||||
easier to have several small boards instead of having one huge board with loads of connectors and cables coming off it.
|
||||
|
||||
The BOM cost per channel for our system is 3ct for a reasonable MOSFET, about 1ct for one eighth of a shift register
|
||||
plus less than a cent for one resistor between shift register and MOSFET. In the end, the connectors are more expensive
|
||||
than the driving circuitry.
|
||||
|
||||
.. _MOSFET: https://en.wikipedia.org/wiki/MOSFET
|
||||
.. _STM32F030F4P6: http://www.st.com/resource/en/datasheet/stm32f030f4.pdf
|
||||
|
||||
Hardware design
|
||||
===============
|
||||
|
||||
From this starting point, we made a very prototype-y hardware design for a 32-channel 12V LED tape driver. The design is
|
||||
based on the STM32F030F4P6_ driving the shift registers as explained above. The system is controlled through an RS485_
|
||||
bus that is connected up to the microcontroller's UART using an MAX485_-compatible RS485 transceiver. The LED tape is
|
||||
connected using 9-pin SUB-D_ connectors since they are cheap and good enough for the small current of our short segments
|
||||
of LED tape. The MOSFETs we use are small SOT-23_ logic-level MOSFETs. In various prototypes we used both International
|
||||
Rectifier's IRLML6244_ as well as Alpha & Omega Semiconductor's AO3400_. Both are good up to about 30V/5A. Since we're
|
||||
only driving about 2m of LED tape per channel we're not going above about 0.5A and the MOSFETs don't even get warm.
|
||||
|
||||
.. _RS485: https://en.wikipedia.org/wiki/RS-485
|
||||
.. _MAX485: https://datasheets.maximintegrated.com/en/ds/MAX1487-MAX491.pdf
|
||||
.. _IRLML6244: https://www.infineon.com/dgdl/?fileId=5546d462533600a4015356686fed261f
|
||||
.. _AO3400: http://aosmd.com/pdfs/datasheet/AO3400.pdf
|
||||
.. _SUB-D: https://en.wikipedia.org/wiki/D-subminiature
|
||||
.. _SOT-23: http://www.nxp.com/documents/outline_drawing/SOT23.pdf
|
||||
|
||||
Switching nonlinearities
|
||||
------------------------
|
||||
During testing of our initial prototype, we noticed that the brightness seemed to jump around when fading to very low
|
||||
values. It turned out that our extremely simple LED driving circuit consisting of only the shift register directly
|
||||
driving a MOSFET, which in turn directly drives the LED tape was maybe a little bit too simple. After some measurements
|
||||
it turned out that we were looking at about 6Vpp of ringing on the driver's output voltage. The picture below is the
|
||||
voltrage we saw on our oscilloscope on the LED tape.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/driver_ringing_strong.jpg" alt="Strong ringing on the LED voltage waveform edge at about
|
||||
100% overshoot during about 70% of the cycle time.">
|
||||
<figcaption>Bad ringing on the LED output voltage caused by wiring inductance. Note that the effect on the
|
||||
actual LED current is less bad than this looks since the LED's V/I curve is nonlinear.</figcaption>
|
||||
</figure>
|
||||
|
||||
|
||||
Dynamic switching behavior: Cause and Effect
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
A bit of LTSpice_ action later we found that the inductance of the few metres of cable leading to the LED tape is the
|
||||
likely culprit. The figure below is the schematic used for the simulations.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/driver_output_ltspice_schematic.jpg" alt="The LTSpice schematic of one output of the driver,
|
||||
taking into account the shift register's output ESR and the wiring ESL.">
|
||||
<figcaption>The schematic of the simulation in LTSpice</figcaption>
|
||||
</figure>
|
||||
|
||||
As tested, the driver does not include any per-output smoothing so the ~.5A transient on each BCM cycle hits the cable
|
||||
in full. Combined with the cable inductance, this works out to a considerable lag of the rising edge of the LED
|
||||
current, and bad ringing on its falling edge. Below is the voltage on the LED output from an LTSpice simulation of our
|
||||
driver.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/overshoot_sim_r0.svg" alt="The result of the LTSpice simulation of our driver output. The LED
|
||||
current shows similar ringing to what we measured using the oscilloscope. Interestingly, the gate voltage shows
|
||||
strong ringing, too.">
|
||||
<figcaption>The result of our LTSpice simulation. This simulation assumes 1µH of wiring inductance and 50Ω of
|
||||
output impedance on the part of the shift register. The ringing at the gate visible in the gate voltage graph is
|
||||
due to feed-through of the ringing at the output through the MOSFET's parasitic Cgd.</figcaption>
|
||||
</figure>
|
||||
|
||||
We were able to reduce the rining and limit the effect somewhat by putting a 220Ω series resistor in between the shift
|
||||
register output and the MOSFET gate. This resistor forms an RC circuit with the MOSFET's nanofarad or two of gate
|
||||
capacitance. The result of this is that the LED current passing the wire's ESL rises slightly more slowly and thus the
|
||||
series inductance gets excited slightly less, and the overshoot decreases. Below is a picture of the waveform with the
|
||||
damping resistor in place and a picture of our measurement for comparison. The resistor values don't agree perfectly
|
||||
since the estimated ESL and stray capacitance of the wiring is probably way off.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/driver_ringing_weak.jpg" alt="Weak ringing on the LED voltage waveform edge at about 30%
|
||||
overshoot during about 20% of the cycle time.">
|
||||
<figcaption>Adding a resistor in front of the MOSFET gate to slow the transition damped the ringing somewhat,
|
||||
but ultimately it cannot be eliminated entirely. Note how you can actually see the miller plateau on the
|
||||
trailing edge of this signal.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/overshoot_sim_r100.svg" alt="The result of the LTSpice simulation of our driver output with an
|
||||
extra 100 Ohms between shift register output and MOSFET gate. Similar to the oscilloscope measurement the
|
||||
ringing is much reduced in its amplitude.">
|
||||
<figcaption>The LTSpice simulation result with the same parameters as above but with an extra 100Ω between the
|
||||
shfit register's output and the MOSFET's gate.</figcaption>
|
||||
</figure>
|
||||
|
||||
A side effect of this fix is that now the effective on-time of the LED tape is much longer than the duty cycle at the
|
||||
shift register's output at very small duty cycles (1µs or less). This is caused by the MOSFET's `miller
|
||||
plateau`_. For illustration, below is a graph of both the excitation waveform (the boxy line) and the resulting LED
|
||||
current (the other ones) both without damping (top) and with 220Ω damping (bottom). As you can see the effective duty
|
||||
cycle of the LED current is not at all equal to the 50% duty cycle of the excitation square wave.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/asymmetric_iled.svg" alt="The result of an LTSpice simulation of the LED duty cycle without and
|
||||
with damping. Dampening widens the LED current waveform from 50% duty cycle with sharp edges to about 80% duty
|
||||
cycle with soft edges.">
|
||||
<figcaption>Simulated LED duty cycle with and without damping. The damping resistance used in this simulation
|
||||
was 220Ω.</figcaption>
|
||||
</figure>
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/asymmetric_vgate.svg" alt="The gate voltages in the spice simulation above. The undamped
|
||||
response shows sharp edges with the miller plateau being a barely noticeable step, but with strong ringing on
|
||||
the trailing edge. The damped response shows RC-like slow-edges, but has wide miller plateaus on both edges
|
||||
adding up to about 50% of the pulse width.">
|
||||
<figcaption>The MOSFET gate voltage from the simulation in the figure above. You can clearly see how the miller
|
||||
plateau (the horizontal part of the trace at about 1V) is getting much wider with added damped, and how the
|
||||
resulting gate charge/discharge curve is not at all that of a capacitor anymore.</figcaption>
|
||||
</figure>
|
||||
|
||||
|
||||
|
||||
In conclusion, we have three major causes for our calculated LED brightness not matching reality:
|
||||
|
||||
* Ringing of the equivalent series inductance of the wiring leading up to the LED tape
|
||||
* Miller plateau lag
|
||||
* The damping resistor and the MOSFET gate forming an RC filter that helps with wire ESL ringing but worsens the miller
|
||||
plateau issue and deforms the LED current edges.
|
||||
|
||||
Added up, these three effects yield a picture that agrees well with our simulations and measurements. The overall effect
|
||||
is neglegible at long period durations (>10µs), but gets really bad at short period durations (<1µs). The effect is
|
||||
non-linear, so correcting for it is not as simple as adding an offset.
|
||||
|
||||
.. _LTSpice: http://www.analog.com/en/design-center/design-tools-and-calculators/ltspice-simulator.html
|
||||
.. _`miller plateau`: https://www.vishay.com/docs/68214/turnonprocess.pdf
|
||||
|
||||
Measuring LED tape brightness
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
In order to correct for the nonlinearities mentioned above, we decided to implement a lookup table mapping BCM period to
|
||||
actual timer setting. That is, each row of the table contains the actual period length we need to set the
|
||||
microcontroller's timer to in order to get our intended brightness steps.
|
||||
|
||||
To calibrate our driver, we needed a setup for reproducible measurement of the relative brightness of our LED tape at
|
||||
different settings. Absolute brightness is not of interest to us as the eye can't perceive it. To perform the
|
||||
calibration, the LED driver is set to enable each single BCM period in turn, i.e. brightness values 1, 2, 4, 8, 16 etc.
|
||||
|
||||
The setup we used to measure the LED tape's brightness consists of a bunch of LED tape stuck into a tin can for
|
||||
shielding against both stray light and electromagnetic interference and a photodiode looking at the LED tape. We used
|
||||
the venerable BPW34_ photodiode in our setup as I had a bunch leftover from another project and because they are quite
|
||||
sensitive owing to their physically large die area.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/linearization_setup.jpg" alt="The led measurement setup consists of several PCBs and a
|
||||
breadboard linked with a bunch of wires and a big tin can to shield the LEDs and the photodiode. A large sub-D
|
||||
connector is put into the top of the tin can as a feed-through for the LED tape's control signals and the
|
||||
photodiode signal. In the background the control laptop is visible.">
|
||||
<figcaption>The LED brighness measurement setup. The big tin can contains a bunch of LED tape and the
|
||||
photodiode. The breadboard on the right is used for the photodiode preamplifier and for jumpering around the LED
|
||||
tape's channels. The red board next to it is the buspirate used as ADC. The board on the bottom left is a
|
||||
TTL-to-RS485 converter and the board in the middle is the unit under test.</figcaption>
|
||||
</figure>
|
||||
|
||||
The photodiode's photocurrent is converted into a voltage using a very simple transimpedance amplifier based around a
|
||||
MCP6002_ opamp that was damped into oblivion with a couple nanofarads of capacitance in its feedback loop. The MCP6002_
|
||||
is a fine choice here since I had a bunch and because it is a CMOS opamp, meaning it has low bias current that would
|
||||
mess up our measurements. For many applications, opamp bias current is not a big issue but when using the opamp to
|
||||
directly measure very small currents at its input it quickly swamps out the signal for most BJT-input types.
|
||||
|
||||
The transimpedance amplifier's output is read from the computer using the ADC input of a buspirate USB thinggamajob. In
|
||||
general I would not recommend the buspirate as a tool for this job since it's ADC is not particularly good and it's
|
||||
programming interface is positively atrocious, but it was what I had and it beat first wiring up one of the dedicated
|
||||
ADC chips I had in my parts bin.
|
||||
|
||||
The computer runs a small python script cycling the LED tape through all its BCM period settings and taking a brightness
|
||||
measurement at each step. Later on, these measurements can be plotted to visualize the resulting slope's linearity, and
|
||||
we can even do a simulation of the resulting brightness for all possible control values by just adding the measured
|
||||
photocurrents for a certain BCM setpoint just as our retinas would do.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/driver_linearity_raw.svg" alt="">
|
||||
<figcaption>
|
||||
A plot of the measured brightness of our LED tape for each BCM period. The brightness values are normalized
|
||||
to the value measured at the LSB setpoint (brightness=1/65535). Ideally, this plot would show a straight
|
||||
line with slope 1. Obviously, it doesn't. The bend in the curve is caused by the above-mentioned duty cycle
|
||||
offset adding an offset to all brightness values. Shown is both the raw data (light), which has essentially zero
|
||||
measurement error and a linear fit (dark).
|
||||
|
||||
The plot is in log-log to approximate how the human eye would perceive brightness, i.e. highly sensitive at
|
||||
low values but not very sensitive at all at large values.
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
While it would be possible to fully automate the optimization of BCM driver lookup tables, we needed only one and in the
|
||||
end I just sat down and manually tweaked the ideal values we initially calculated until I liked the result. You can see
|
||||
the resulting brightness curve below.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<figure class="side-by-side">
|
||||
<img src="images/uncorrected_brightness_sim.svg" alt="">
|
||||
<figcaption>
|
||||
Calculated brightness curve for the uncorrected BCM setup. As you can see, at low setpoints the result
|
||||
is about as smooth as sandpaper, which is well in line with our observations. At high setpoints the
|
||||
offset gets swamped out and the nonlinearity in the low bits is not visible anymore.
|
||||
</figcaption>
|
||||
</figure><figure class="side-by-side">
|
||||
<img src="images/corrected_brightness_sim.svg" alt="">
|
||||
<figcaption>
|
||||
Brightness curve for the corrected BCM setup extrapolated using actual measurements. Looks as buttery
|
||||
smooth in real life as it does in this plot.
|
||||
</figcaption>
|
||||
</figcaption>
|
||||
</figure>
|
||||
</figure>
|
||||
|
||||
.. _BPW34: http://www.vishay.com/docs/81521/bpw34.pdf
|
||||
.. _MCP6002: http://ww1.microchip.com/downloads/en/DeviceDoc/21733j.pdf
|
||||
|
||||
Controlling the driver
|
||||
----------------------
|
||||
|
||||
Now that our driver was behaving linear enough that you couldn't see it actually wasn't we needed a nice way to control
|
||||
it from a computer of our choice. In the ultimate application (our staircase) we'll use a raspberry pi for this. Since
|
||||
we already settled on an RS485_ bus for its robustness and simplicity, we had to device a protocol to control the driver
|
||||
over this bus. Here, we settled on a simple, COBS_-based protocol for the reasons I wrote about in `How to talk to your
|
||||
microcontroller over serial <serial-protocols>`_.
|
||||
|
||||
To address our driver nodes, we modified the Makefile to build a random 32-bit MAC into each firmware image. The
|
||||
protocol has only five message types:
|
||||
|
||||
1. A 0-byte *ping* packet, to which each node would reply with its own address in the
|
||||
first 100ms after boot. This can be used to initially discover the addresses of all nodes connected to the bus. You'd
|
||||
spam the bus with *ping* packets, and then hit reset on each node in turn. The control computer would then receive
|
||||
each device's MAC address as you hit reset.
|
||||
2. A 4-byte *address* packet that says which device that the following packet is for. This way of us using the packet
|
||||
length instead of a packet type field is not particularly elegant, but our system is simple enough and it was easy to
|
||||
implement.
|
||||
3. A 64-byte *frame buffer* packet that contains 16 bits of left-aligned brightness data for every channel
|
||||
4. A one-byte *get status* packet that tells the device to respond with...
|
||||
5. ...a 27-byte status packet containing a brief description of the firmware (version number, channel count, bit depth
|
||||
etc.) as well as the device's current life stats (VCC, temperature, uptime, UART frame errors etc.).
|
||||
|
||||
Wrapped up in a nice python interface we can now easily enumerate any drivers we connect to a bus, query their status
|
||||
and control their outputs.
|
||||
|
||||
.. _COBS: https://en.wikipedia.org/wiki/Consistent_Overhead_Byte_Stuffing
|
||||
|
||||
Conclusion
|
||||
----------
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<figure class="side-by-side">
|
||||
<a href="images/olsndot_schematic.png">
|
||||
<img src="images/olsndot_schematic.png" alt="A picture of the LED driver schematic">
|
||||
</a>
|
||||
<figcaption>The LED driver <a href="images/olsndot_schematic.png">schematic</a></figcaption>
|
||||
</figure><figure class="side-by-side">
|
||||
<a href="images/olsndot_pcb.png">
|
||||
<img src="images/olsndot_pcb.png" alt="A picture of the LED driver PCB layout">
|
||||
</a>
|
||||
<figcaption>The LED driver <a href="images/olsndot_pcb.png">PCB layout</a></figcaption>
|
||||
</figure>
|
||||
</figure>
|
||||
|
||||
Putting some thought into the control circuitry and software, you can easily control large numbers of channels of LEDs
|
||||
using extremely inexpensive driving hardware without any compromises on dynamic range. The design we settled on can
|
||||
drive 32 channels of LED tape with a dynamic range of 14bit at a BOM cost of below 10€. All it really takes is a couple
|
||||
of shift registers and a mildly bored STM32 microcontroller.
|
||||
|
||||
Get a PDF file of the schematic and PCB layout `here <olsndot_v02_schematics_and_pcb.pdf>`__ or download the CAD files
|
||||
and the firmware sources `from github <https://github.com/jaseg/led_drv>`_. You can view the Jupyter notebook used to
|
||||
analyze the brightness measurement data `here <http://nbviewer.jupyter.org/github/jaseg/led_drv/blob/master/doc/Run_analysis.ipynb>`__.
|
||||
|
||||
|
|
@ -1,38 +0,0 @@
|
|||
---
|
||||
title: "Private Contact Discovery"
|
||||
date: 2019-06-22T10:30:00+08:00
|
||||
---
|
||||
|
||||
Private Contact Discovery
|
||||
=========================
|
||||
|
||||
Private Contact Discovery (PCD) is the formal name for the problem modern smartphone messenger applications have on
|
||||
installation: Given a user's address book, find out which of their contacts also use the same messenger without the
|
||||
messenger's servers learning anything about the user's address book. The widespread non-private way to do this is to
|
||||
simply upload the user's address book to the app's operator's servers and do an SQL JOIN keyed on the phone number field
|
||||
against the database of registered users. People have tried sprinkling some hashes over these phone numbers in an
|
||||
attempt to improve privacy, but obviously running a brute-force preimage attack given a domain of maybe a few billion
|
||||
valid inputs is not cryptographically hard.
|
||||
|
||||
Private Contact Discovery can be phrased in terms of Private Set Intersection (PSI), the cryptographic problem of having
|
||||
two parties holding one set each find the intersection of their sets without disclosing any other information. PSI has
|
||||
been an active field of research for a while and already yielded useful results for some use cases. Alas, none of those
|
||||
results were truly practical yet for usage in PCD in a typical messenger application. They would require too much CPU
|
||||
time or too much data to be transferred.
|
||||
|
||||
At USENIX Security 2019, Researchers from technical universities Graz and Darmstadt published a paper titled *Private
|
||||
Contact Discovery at Scale*
|
||||
(`eprint <https://eprint.iacr.org/2019/517>`__ | `PDF <https://eprint.iacr.org/2019/517.pdf>`__).
|
||||
In this paper, they basically optimize the hell out of existing cryptographic solutions to private contact discovery,
|
||||
jumping from a still-impractical state of the art right to practicality. Their scheme allows a client with 1k contacts
|
||||
to run PCD against a server with 1B contacts in about 3s on a phone. The main disadvantage of their scheme is that it
|
||||
requires the client to in advance download a compressed database of all users, that clocks in at about 1GB for 1B users.
|
||||
|
||||
I found this paper very interesting for its immediate practical applicability. As an excuse to dig into the topic some
|
||||
more, I gave a short presentation at my university lab's research seminar on this paper
|
||||
(slides: `PDF <mori_semi_psi_talk.pdf>`__ | `ODP <mori_semi_psi_talk.odp>`__).
|
||||
|
||||
Even if you're not working on secure communication systems on a day-to-day basis this paper might interest you. If
|
||||
you're working with social account information of any kind I can highly recommend giving it a look. Not only might your
|
||||
users benefit from improved privacy, but your company might be able to avoid a bunch of data protection and
|
||||
accountability issues by simply not producing as much sensitive data in the first place.
|
||||
|
|
@ -1,249 +0,0 @@
|
|||
---
|
||||
title: "How to talk to your microcontroller over serial"
|
||||
date: 2018-05-19T08:09:46+02:00
|
||||
---
|
||||
|
||||
Scroll to the end for the `TL;DR <Conclusion_>`_.
|
||||
|
||||
In this article I will give an overview on the protocols spoken on serial ports, highlighting common pitfalls. I will
|
||||
summarize some points on how to design a serial protocol that is simple to implement and works reliably even under error
|
||||
conditions.
|
||||
|
||||
If you have done low-level microcontroller firmware you will regularly have had to stuff some data up a serial port to
|
||||
another microcontroller or to a computer. In the age of USB, a serial port is still the simplest and quickest way to get
|
||||
communication to a control computer up and running. Integrating a ten thousand-line USB stack into your firmware and
|
||||
writing the necessary low-level drivers on the host side might take days. Poking a few registers to set up your UART to
|
||||
talk to an external hardware USB to serial converter is a matter of minutes.
|
||||
|
||||
This simplicity is treacherous, though. Oftentimes, you start writing your serial protocol as needs arise. Things might
|
||||
start harmless with something like ``SET_LED ON\n``, but unless you proceed it is easy to end up in a hot mess of command
|
||||
modes, protocol states that breaks under stress. The ways in which serial protocols break are manifold. The simplest one
|
||||
is that at some point a character is mangled, leading to both ends of the conversation ending up in misaligned protocol
|
||||
states. With a fragile protocol, you might end up in a state that is hard to recover from. In extreme cases, this leads
|
||||
to code such as `this gem`_ performing some sort of arcane ritual to get back to some known state, and all just because
|
||||
someone did not do their homework. Below we'll embark on a journey through the lands of protocol design, exploring the
|
||||
facets of this deceptively simple problem.
|
||||
|
||||
.. _`this gem`: https://github.com/juhasch/pyBusPirateLite/blob/master/pyBusPirateLite/BBIO_base.py#L68
|
||||
|
||||
Text-based serial protocols
|
||||
===========================
|
||||
|
||||
The first serial protocol you've likely written is a human-readable, text-based one. Text-based protocols have the big
|
||||
advantage that you can just print them on a terminal and you can immediately see what's happening. In most cases you can
|
||||
even type out the protocol with your bare hands, meaning that you don't really need a debugging tool beyond a serial
|
||||
console.
|
||||
|
||||
However, text-based protocols also have a number of disadvantages. Depending on your application, these might not matter
|
||||
and in many cases a text-based protocol is the most sensible solution. But then, in some cases they might and it's good
|
||||
to know when you hit one of them.
|
||||
|
||||
Problems
|
||||
--------
|
||||
|
||||
Low information density
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
Generally, you won't be able to stuff much more than four or five bit of information down a serial port using a
|
||||
human-readable protocol. In many cases you will get much less. If you have 10 commands that are only issued a couple
|
||||
times a second nobody cares that you spend maybe ten bytes per command on nice, verbose strings such as ``SET LED``. But
|
||||
if you're trying to squeeze a half-kilobyte framebuffer down the line you might start to notice the difference between
|
||||
hex and base-64 encoding, and a binary protocol might really be more up to the job.
|
||||
|
||||
Complex parsing code
|
||||
~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
On the computer side of thing, with the whole phalanx of an operating system, the standard library of your programming
|
||||
language of choice and for all intents and purposes unlimted CPU and memory resources to spare you can easily parse
|
||||
anything spoken on a serial port in real time, even at a blazing fast full Megabaud. The microcontroller side however is
|
||||
an entirely different beast. On a small microcontroller, printf_ alone will eat about half your flash. On most small
|
||||
microcontrollers, you just won't get a regex library even though it would make parsing textual commands *so much
|
||||
simpler*. Lacking these resources, you might end up hand-knitting a lot of low-level C code to do something seemingly
|
||||
simple such as parsing ``set_channel (13, 1.1333)\n``. These issues have to be taken into account in the protocol design
|
||||
from the beginning. If you don't really need matching parentheses, don't use them.
|
||||
|
||||
Fragile protocol state
|
||||
~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
Say you have a ``SET_DISPLAY`` command. Now say your display can display four lines of text. The obvious approach to this
|
||||
is probably the SMTP_ or HTTP_ way of sending ``SET_DISPLAY\nThis is line 1\nThis is line 2\n\n``. This would certainly
|
||||
work, but it is very fragile. With this protocol, you're in trouble if at any point the terminating second newline
|
||||
character gets mangled (say, someone unplugs the cable, or the control computer reboots, or a cosmic ray hits something
|
||||
and ``0x10 '\n'`` turns into ``0x50 'P'``).
|
||||
|
||||
.. _SMTP: https://en.wikipedia.org/wiki/Simple_Mail_Transfer_Protocol
|
||||
.. _HTTP: https://en.wikipedia.org/wiki/Hypertext_Transfer_Protocol
|
||||
|
||||
Timeouts don't work
|
||||
~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
You might try to solve the problem of your protocol state machine tangling up with a timeout. "If I don't get a valid
|
||||
command for more than 200ms I go back to default state." But consider the above example. Say, your control computer
|
||||
sends a ``SET_DISPLAY`` command every 100ms. If in one of them the state machine tangles up, the parser hangs since the
|
||||
timeout is never hit, a new line of text arriving every 100ms.
|
||||
|
||||
Framing is hard
|
||||
~~~~~~~~~~~~~~~
|
||||
|
||||
You might also try to drop the second newline and using a convention such as ``SET_DISPLAY`` is followed by two lines of
|
||||
text, then commands resume.". This works as long as your display contents never look like commands. If you are only ever
|
||||
displaying the same three messages on a character LCD that might work, but if you're displaying binary framebuffer
|
||||
data you've lost.
|
||||
|
||||
Solutions
|
||||
---------
|
||||
|
||||
Keep the state machine simple
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
Always use a single line of text to represent a single command. Don't do protocol states or modes where you can toggle
|
||||
between different interpretations for a line. If you have to send human-readable text as part of a command (such as
|
||||
``SET_DISPLAY``) escape it so it doesn't contain any newlines.
|
||||
|
||||
This way, you keep your protocol state machine simple. If at any time your serial trips and flips a bit or looses a byte
|
||||
your protocol will recover on the next newline character, returning to its base state.
|
||||
|
||||
Encode numbers in hex when possible
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
Printing a number in hexadecimal is a very tidy operation, even on the smalest 8-bit microcontrollers. In contrast,
|
||||
printing decimal requires both division and remainder in a loop which might get annoyingly code- and time-intensive on
|
||||
large numbers (say a 32-bit int) and small microcontrollers.
|
||||
|
||||
If you have to send fractional values, consider their precision. Instead of sending a 12 bit ADC result as a 32-bit
|
||||
float formatted like ``0.176513671875`` sending ``0x2d3`` and dividing by 4096 on the host might be more sensible. If you
|
||||
really have to communicate big floats and you can't take the overhead of including both printf_ and scanf_ you can
|
||||
use hexadecimal floating point, which is basically ``hex((int)foo) + "." + hex((int)(65536*(foo - (int)foo)))`` for four
|
||||
digits. You can also just hex-encode the binary IEEE-754_ representation of the float, sending ``hex(*(int *)&float)``.
|
||||
Most programming languages will have a `simple, built-in means to parse this sort of thing
|
||||
<https://docs.python.org/3.5/library/struct.html>`__.
|
||||
|
||||
.. _printf: http://git.musl-libc.org/cgit/musl/tree/src/stdio/vfprintf.c
|
||||
.. _scanf: http://git.musl-libc.org/cgit/musl/tree/src/stdio/vfscanf.c
|
||||
.. _IEEE-754: https://en.wikipedia.org/wiki/IEEE_754
|
||||
|
||||
Escape multiline strings
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
If you have to send arbitrary strings, escape special characters. This not only has the advantage of yielding a robust
|
||||
protocol: It also ensures you can actually see everything that's going on when debugging. The string ``"\r\n"`` is easy to
|
||||
distinguish from ``"\n"`` while your terminal emulator might not care.
|
||||
|
||||
The simplest encoding to use is the C-style backslash encoding. Host-side, most languages will have a `built-in means of
|
||||
escaping a string like that <https://docs.python.org/3.5/library/codecs.html#text-encodings>`__.
|
||||
|
||||
Encoding binary data
|
||||
--------------------
|
||||
|
||||
For binary data, hex and base-64 are the most common encodings. Since hex is simpler to implement I'd go with it unless
|
||||
I really need the 30% bandwidth improvement base-64 brings.
|
||||
|
||||
Binary serial protocols
|
||||
=======================
|
||||
|
||||
In contrast to anything human-readable, binary protocols are generally more bandwidth-efficient and are easier to format
|
||||
and parse. However, binary protocols come with their own version of the caveats we discussed for text-based protocols.
|
||||
|
||||
The framing problem in binary protocols
|
||||
---------------------------------------
|
||||
|
||||
The most basic problems with binary protocols as with text-based ones is framing, i.e. splitting up the continuous
|
||||
serial data stream into discrete packets. The issue is that it is that you have to somehow mark boundaries between
|
||||
frames. The simplest way would be to use some special character to delimit frames, but then any 8-bit character you
|
||||
could choose could also occur within a frame.
|
||||
|
||||
SLIP/PPP-like special character framing
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
Some protocols solve this problem much like we have solved it above for strings in line-based protocols, by escaping any
|
||||
occurence of the special delimiter character within frames. That is, if you want to use ``0x00`` as a delimiter, you would
|
||||
encode a packet containing ``0xde 0xad 0x00 0xbe 0xef`` as something like ``0xde 0xad 0x01 0x02 0xbe 0xef``, replacing the
|
||||
null byte with a magic sequence. This framing works, but is has one critical disadvantage: The length of the resulting
|
||||
escaped data is dependent on the raw data, and in the worst case twice as long. In a raw packet consisting entirely of
|
||||
null bytes, every byte must be escaped with two escape bytes. This means that in this case the packet length doubles,
|
||||
and in this particular case we're even less efficient than base-64.
|
||||
|
||||
Highly variable packet length is also bad since it makes it very hard to make any timing guarantees for our protocol.
|
||||
|
||||
9-bit framing
|
||||
~~~~~~~~~~~~~
|
||||
|
||||
A framing mode sometimes used is to configure the UARTs to transmit 9-bit characters and to use the 9th bit to designate
|
||||
control characters. This works really well, and gives plenty of control characters to work with. The main problem with
|
||||
this is that a 9-bit serial interface is highly nonstandard and you need UARTs on both ends that actually support this
|
||||
mode. Another issue is that though more efficient than both delmitier-based and purely text-based protocols, it still
|
||||
incurs an extra about 10% of bandwidth overhead. This is not a lot if all you're sending is a little command every now
|
||||
and then, but if you're trying to push large amounts of data through your serial it's still bad.
|
||||
|
||||
COBS
|
||||
~~~~
|
||||
|
||||
Given the limitations of the two above-mentioned framing formats, we really want something better. The `Serial Line
|
||||
Internet Protocol (SLIP)`_ as well as the `Point to Point Protocol (PPP)`_, standardized in 1988 and 1994 respectively,
|
||||
both use escape sequences. This might come as a surprise, but humanity has actually still made significant technological
|
||||
progress on protocols for 8-bit serial interfaces until the turn of the millennium. In 1999, `Consistent Overhead Byte
|
||||
Stuffing (COBS)`_ (`wiki <https://en.wikipedia.org/wiki/Consistent_Overhead_Byte_Stuffing>`__) was published by a few
|
||||
researchers from Apple Computer and Stanford University. As a reaction on the bandwidth doubling problem present in
|
||||
PPP_, COBS *always* has an overhead of a single byte, no matter what or how long a packet's content is.
|
||||
|
||||
COBS uses the null byte as a delimiter interleaves all the raw packet data and a `run-length encoding`_ of the non-zero
|
||||
portions of the raw packet. That is, it prepends the number of bytes until the first zero byte to the packet, plus one.
|
||||
Then it takes all the leading non-zero bytes of the packet, unmodified. Then, it again encodes the distance from the
|
||||
first zero to the second zero, plus one. And then it takes the second non-zero run of bytes unmodified. And so on. At
|
||||
the end, the packet is terminated with a zero byte.
|
||||
|
||||
The result of this scheme is that the encoded packet does not contain any zero bytes, as every zero byte has been
|
||||
replaced with the number of bytes until the next zero byte, plus one, and that can't be zero. Both formatter and parser
|
||||
each have to keep a counter running to keep track of the distances between zero bytes. The first byte of the packet
|
||||
initializes that counter and is dropped by the parser. After that, every encoded byte received results in one raw byte
|
||||
parsed.
|
||||
|
||||
While this might sound more complicated than the escaping explained above, the gains in predictability and efficiency
|
||||
are worth it. An implementation of encoder and decoder should each be about ten lines of C or two lines of Python. A
|
||||
minor asymmetry of the protocol is that while decoding can be done in-place, encoding either needs two passes or you
|
||||
need to scan forward for the next null byte.
|
||||
|
||||
.. _`Point to Point Protocol (PPP)`: https://en.wikipedia.org/wiki/Point-to-Point_Protocol
|
||||
.. _PPP: https://en.wikipedia.org/wiki/Point-to-Point_Protocol
|
||||
.. _`Serial Line Internet Protocol (SLIP)`: https://en.wikipedia.org/wiki/Serial_Line_Internet_Protocol
|
||||
.. _`Consistent Overhead Byte Stuffing (COBS)`: http://www.stuartcheshire.org/papers/COBSforToN.pdf
|
||||
.. _`Point-to-Point Protocol (PPP)`: https://en.wikipedia.org/wiki/Point-to-Point_Protocol
|
||||
.. _`run-length encoding`: https://en.wikipedia.org/wiki/Run-length_encoding
|
||||
|
||||
State machines and error recovery
|
||||
---------------------------------
|
||||
|
||||
In binary protocols even more than in textual ones it is tempting to build complex state machines triggering actions on
|
||||
a sequence of protocol packets. Please resist that temptation. As with textual protocols keeping the protocol state to
|
||||
the minimum possible allows for a self-synchronizing protocol. A serial protocol should be designed such that if due to
|
||||
a dropped packet or two both ends will naturally re-synchronize within another packet or two. A simple way of doing that
|
||||
is to always transmit one semantic command per packet and to design these commands in the most idempotent_ way possible.
|
||||
For example, when filling a framebuffer piece by piece, include the offset in each piece instead of keeping track of it
|
||||
on the receiving side.
|
||||
|
||||
.. _idempotent: https://en.wikipedia.org/wiki/Idempotence#Computer_science_meaning
|
||||
|
||||
Conclusion
|
||||
==========
|
||||
|
||||
Here's your five-step guide to serial bliss:
|
||||
|
||||
1. Unless you have super-special requirements, always use the slowest you can get away with from 9600Bd, 115200Bd or
|
||||
1MBd. 8N1 framing if you're talking to anything but another microcontroller on the same board. These settings are
|
||||
the most common and cover any use case. You'll inevitably have to guess these at some point in the future.
|
||||
2. If you're doing something simple and speed is not a particular concern, use a human-readable text-based protocol. Use
|
||||
one command/reply per line, begin each line with some sort of command word and format numbers in hexadecimal. You get
|
||||
bonus points if the device replies to unknown commands with a human-readable status message and prints a brief
|
||||
protocol overview on boot.
|
||||
3. If you're doing something even slightly nontrivial or need moderate throughput (>1k commands per second or >20 byte of
|
||||
data per command) use a COBS-based protocol. If you don't have a better idea, go for an ``[target MAC][command
|
||||
ID][command arguments]`` packet format for multidrop busses. For single-drop you may decide to drop the MAC address.
|
||||
4. Always include some sort of "status" command that prints life stats such as VCC, temperature, serial framing errors
|
||||
and uptime. You'll need some sort of ping command anyway and that one might as well do something useful.
|
||||
5. If at all possible, keep your protocol context-free across packets/lines. That is, a certain command should always be
|
||||
self-contained, and no command should change the meaning of the next packet or line that is sent. This is really
|
||||
important to allow for self-synchronization. If you really need to break up something into multiple commands, say you
|
||||
want to set a large framebuffer in pieces, do it in a idempotent_ way: Instead of sending something like ``FRAMEBUFFER
|
||||
INCOMING:\n[byte 0-16]\n[byte 17-32]\n[...]\nEND OF FRAME`` rather send ``FRAMEBUFFER DATA FOR OFFSET 0: [byte
|
||||
0-16]\nFRAMEBUFFER DATA FOR OFFSET 17: [byte 17-32]\n[...]\nSWAP BUFFERS\n``.
|
||||
|
||||
|
Before Width: | Height: | Size: 574 KiB |
|
|
@ -1,244 +0,0 @@
|
|||
---
|
||||
title: "Theia Attack Resistance and Digital Identity"
|
||||
date: 2020-09-09T15:00:00+02:00
|
||||
---
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure class="header">
|
||||
<img src="images/succulents.jpg">
|
||||
<figcaption>Photo by <a href="https://unsplash.com/@timbennettcreative">Tim Bennett</a> on <a href="https://unsplash.com/">Unsplash</a></figcaption>
|
||||
</figure>
|
||||
|
||||
|
||||
Theia in Cyberspace
|
||||
===================
|
||||
|
||||
In informatics, the term *distributed system* is used to describe the aggregate behavior of a complex network made up of
|
||||
individual computers. For decades, computer scientists to some success have been trying to figure out how exactly the
|
||||
individual computers that make up such a distributed system need to be programmed for the resulting amalgamation to
|
||||
behave in a predictable, maybe even a desirable way. Though seemingly simple on its surface, this problem has a
|
||||
surprising depth to it that has yielded research questions for a whole field for several decades now. One particular
|
||||
as-of-yet unsolved problem is resistance against *theia attacks* (or "sybil" attacks in older terminology).
|
||||
|
||||
Named after the 1973 book by Flora Rheta Schreiber on dissociative identity disorder, a sybil attack is an
|
||||
attack where one computer in a distributed system pretends to be multiple computers to gain an advantage. From your
|
||||
author's standpoint, naming a type of computer security attack after a medical condition was an unfortunate choice.
|
||||
For this reason this post uses the term *Theia attack* to refer to the same concept. Theia is a greek godess of light
|
||||
and glitter and the name alludes to the attacker performing something alike an optical illusion, causing the attacked
|
||||
to perceive multiple distinct images that in the end are all only reflections of the same attacker.
|
||||
|
||||
The core insight of computer science research on theia attacks is that there cannot be any technological way of
|
||||
preventing such an attack, and any practical countermeasure must be grounded in some authority or ground truth that is
|
||||
external to the systems—bridging from technology to its social or political context.
|
||||
|
||||
Looking around, we can see a parallel between this question ("which computer is a real computer?") and a social issue
|
||||
that recently has been growing in importance: Just like computers can pretend to be other computers, they can also
|
||||
pretend to be humans. As can humans. Be it within the context of election manipulation or down-to-earth astroturfing_
|
||||
the recurring issue is that in today's online communities, it is hard for an individual to tell who of their online
|
||||
acquaintances are who they seem to be. Different platforms attempt different solutions to this problem, and all fail in
|
||||
some way or another. Facebook employs good old snitching, turning people against each other and asking them "Do you know
|
||||
this person?". Twitter is more laid-back and avoids this Stasi_ methodology in favor of requiring a working mobile phone
|
||||
number from its subjects, essentially short-circuiting identity verification to the phone company's check of their
|
||||
subscriber's national passport.
|
||||
|
||||
.. the preceding is a simplified representation of these platform's practices. In particular facebook uses several
|
||||
methods depending on the case. I think this abbreviated discussion should be ok for the sake of the argument. I am
|
||||
not 100% certain on the accuracy on the accuracy of the statement though. Does fb still do the snitching thing? Is
|
||||
twitter usually content with a phone number?
|
||||
|
||||
Trusting Crypto-Anarchist Authorities
|
||||
=====================================
|
||||
|
||||
Beyond these centralistic solutions to the problem, crypto-anarchists and anarcho-capitalists have been brewing on some
|
||||
interesting novel approaches to online identity based on *blockchain* distributed ledger technology. Distributed
|
||||
ledgers are a distributed systems design pattern that yields a system that works like an append-only logbook.
|
||||
Participants can create new entries in this logbook, but no one—neither the original author, nor other participants—can
|
||||
retroactively change a logbook entry once it has been written. In the blockchain model, past entries are essentially
|
||||
written into stone. This near-perfect immutability is what opens them for a number of use cases from cryptographic
|
||||
pseudo-currencies [#cryptocurrency]_.
|
||||
|
||||
An overview over a variety of these unconventional blockchain identity verification approaches can be found in `this
|
||||
unpublished 2020 survey by Siddarth, Ivliev, Siri and Berman <https://arxiv.org/ftp/arxiv/papers/2008/2008.05300.pdf>`_.
|
||||
They walk their readers through a number of different projects that try to solve the question "Is this human who they
|
||||
pretend to be?" using joint socio-technological approaches. In the following few sections, you may find a short outline
|
||||
of a small selection of them. The conlusion of this post will be a commentary on these approaches, and on the underlying
|
||||
problem of identity in a digital world.
|
||||
|
||||
.. BrightID
|
||||
|
||||
In one scheme, identity is determined by "notary" computers that aggregate large amounts of information on a user's
|
||||
social contacts. These computers then run an algorithm derived from the SybilGuard_, SybilLimit_ and SybilInfer_ lineage
|
||||
of random-walk based algorithms. These algorithms assume that authentic social graphs are small world graphs: Everyone
|
||||
knows everyone else through a friend's friend's friend. They also assume that there is an upper bound on how many
|
||||
connections with authentic users an attacker can forge: Anyone who is not embedded into the graph well enough is cut
|
||||
out. Like this, they put an upper limit on the number of theia identites an attacker can assume given a certian number
|
||||
of connections to real people.
|
||||
|
||||
Disregarding the catastrophic privacy issues of storing large amounts of data on social relationships on someone else's
|
||||
computer, this second assumption is where this model unfortunately breaks down. Applying common sense, it is completely
|
||||
realistic for an attacker to forge a large number of social connections: This is precisely what most of social media
|
||||
marketing is about! A more malicious angle on this would be to consider how in meatspace [#meatspacefn]_ multi-level
|
||||
marketing schemes are successful in coaxing people to abuse their social graphs to disastrous consequences to the
|
||||
well-being of themselves and others. Similar schemes would certainly be possible in cyberspace as well. An additional
|
||||
point to consider is that the upper limit SybilGuard_ and others place on the number of fake identities one can have is
|
||||
simply not that strict at all. An attacker could still get away with a reasonable number of false identities before
|
||||
getting caught by any such algorithm.
|
||||
|
||||
.. Duniter
|
||||
|
||||
In another scheme, identity is awarded to anyone who can convince several people already in the network to vouch for
|
||||
them, and who is at most a few degrees removed from one of several pre-determined celebrities. Apart from again being
|
||||
vulnerable to conmen and other scammers, this system has the glaring flaw of roundly refusing to recognize any person
|
||||
who is not willing or able to engage with multiple of its members. Along with the system's informal requirement for
|
||||
members to only vouch for people they have physically met this leads to a nonstarter in a cyberspace that grown
|
||||
specifically *because* it transcends national borders and physical distance—two most serious obstacles to in-person
|
||||
communication.
|
||||
|
||||
.. Idena Network
|
||||
|
||||
The last scheme I will outline in this post is based around a set of `Turing tests`_; that is, quizzes that are designed
|
||||
to tell apart man and machine. In this system, all participants have to simultaneously undergo a Turing test once in a
|
||||
fortnight. The idea is that this limits the number of theia identities an attacker can assume since they can only solve
|
||||
that many Turing tests at the same time. The system uses a particular type of picture classification-based Turing test
|
||||
and does not seem to be designed with the blind or mentally disabled in mind with accessibility concerns nowhere to be
|
||||
found in the so-called "manifesto" published by its creators. But even ignoring that, the system obviously fails at an
|
||||
even more basic level: The idea that everyone takes a Turing test at the same time only works in a world without time
|
||||
zones. Or jobs for that matter. Also, it assumes that an attacker cannot simply hire a small army of people someplace
|
||||
else to fool the system.
|
||||
|
||||
.. _SybilLimit: https://www.comp.nus.edu.sg/~yuhf/yuh-sybillimit.pdf
|
||||
.. _SybilGuard: http://www.math.cmu.edu/~adf/research/SybilGuard.pdf
|
||||
.. _SybilInfer: https://www.princeton.edu/~pmittal/publications/sybilinfer-ndss09.pdf
|
||||
.. _`Turing Tests`: https://en.wikipedia.org/wiki/Turing_test
|
||||
|
||||
Identity between Cyberspace and Meatspace
|
||||
=========================================
|
||||
|
||||
A common thread in these solutions, from the Facebook'esque Stasi_ methods to the crypto-anarchist challenge-response
|
||||
utopias, is that they all approach digital identity as a question of Objective Truth™ that can unanimously be decided at
|
||||
a system level—or that can be externalized to the next larger system such as the state. Alas, the important question
|
||||
remains unasked:
|
||||
|
||||
What *is* identity?
|
||||
|
||||
The answer to this question certainly depends on the system being examined. For example, an important reason the
|
||||
capitalist corporations mentioned above require knowledge about their users' identity is to generate plausible
|
||||
statistics for the advertisers that form their customer base, similar to how a farmer will keep statics on yield and
|
||||
quality for the buyers of his crop. With this background, a full decoupling of platform accounts from a notion of legal
|
||||
identity seems at odds with the platform's business model—and we will have to adjust our expectations for reform
|
||||
accordingly.
|
||||
|
||||
A common thread among all systems mentioned above is that they all have a social component to them. For this common use
|
||||
case of social systems, I want to make a suggestion on how we can approach digital identity in a more practical, less
|
||||
discriminatory [#discriminatory]_ manner than any of the methods we discussed above. I think both using people's social
|
||||
connections and proxying the decisions of external authorities such as the state are bad systems to decide who is a
|
||||
person and who is not. I will now illustrate this point a bit. Let us think about how many digital identities a human
|
||||
beign might have. First, consider the case of n=0, someone who simply wants no business with the system at all. For
|
||||
simplicity, let us assume that we have solved this issue of consent, i.e. every person who is identified by the system
|
||||
consents to this practice. For n=1, the approaches outlined above all provide some approximate solution. States may not
|
||||
grant every human sufficient ID (e.g. children, the mentally disabled or prisoners might be left out), and the social
|
||||
systems might fail to catch people who simply do not have any friends, but otherwise their approximations hold. Maybe.
|
||||
But what about n=2, n=3, ...? None of these systems adequately consider cases where a human being might legitimately
|
||||
wish to hold multiple digital identities, non-maliciously.
|
||||
|
||||
Consider a hypothetical lesbian, conservative politician. An active social media presence is a core component of a
|
||||
modern politician's carreer. At the same time, "conservative homophobe" is still well within the realm of tautology and
|
||||
it would be legitimate for this politician to wish to not disclose a large fraction of their private life to the world
|
||||
at large. They might have a separate online identity for matters related to it. For this politician, the social
|
||||
relationship-based systems referenced above would either incorporate outing as a design feature, or they would force
|
||||
the politician to choose either of their two identities: To choose between private life and carreer. When deferring to
|
||||
the state as the decider over personhood, at least the platform's operator would know about the outrageously sensitive
|
||||
link between the politician's online identities. Clearly, no such solution can be considered socially just.
|
||||
|
||||
Let us try not to be caught up on saving the world at this point. The issue of conservative homophobia is out of the
|
||||
scope of our consideration, and it is not one that anyone can solve in the near future. Magical realism aside, least of
|
||||
all can some technological thing beckon this change. There is a case for legitimate uses of multiple, separate digital
|
||||
identities, and we do not have a technical or political answer to it. All hope is not lost yet, though. We can easily
|
||||
undo this gordian knot by acknowledging an unspoken assumption that underlies any social relationships between real
|
||||
people, past the procrustean bed of computer systems or organizational structures these relationships are cast into.
|
||||
|
||||
As a function of social interaction, digital identities conform to roles_ in sociological terminology, and are not
|
||||
at all the same as personhood_. Roles are subjective and arise from a relationship between people, and a single
|
||||
person might legitimately perform different roles depending on context.
|
||||
|
||||
When computer scientists or programmers are creating new systems, there always is an (often implicit) modelling stage.
|
||||
Formally, during this stage a domain expert and a modeller with a computer science background come together, each
|
||||
contributing their knowledge to form a model that is both appropriate for real-world use and practical from an
|
||||
engineering point of view. In practice, these two roles are often necessarily fulfilled by the same person, who is often
|
||||
also the programmer of the thing. This leads to many computer systems using poor models. A typical example of this issue
|
||||
are systems requiring a person's name that use three input fields labelled "First Name", "Middle Initial" and "Last
|
||||
Name". These systems are often created by US-American programmers, who are used to this naming schema from their lived
|
||||
experience. Unfortunately, this schema breaks down for those few billion people who use their last name first, who have
|
||||
more than one middle name, or who have multiple given names and do not normally use the first one of those.
|
||||
|
||||
Once a system creator's implicit assumptions have been encoded into the system like this, it is often very hard to get
|
||||
out of that situation. A pattern to use during careful modelling is to keep the model flexible to account for unforeseen
|
||||
corner cases. For example, when modelling a system requiring a person's name, one would have to ask what the name is
|
||||
used for. It may be the most sensible decision to simply ask the user for their name twice: Once in first name/last name
|
||||
format for e.g. tax purposes, and once with a free-form text field for e.g. displaying on their account page.
|
||||
|
||||
While for names, many systems already use some form of flexible model by e.g. having a *handle* or *nickname* separate
|
||||
from the *display name*, "social" systems still often are stuck with an identity model based around a concept of a
|
||||
single, rigid identity. In practice, people perform different roles_ in different circumstances. When asking for a
|
||||
person's identity, one would get wildly different answers from different people. A person's identity as perceived by
|
||||
others is coupled to their relationship more than to some underlying, biological or administrative truth. Thinking back
|
||||
to the straw man politician above, this is evident in subtle ways in almost all our everyday relationships: Some people
|
||||
may know me by my legal name, some by my online nickname. To some I may be a computer scientist, to some a flatmate.
|
||||
None of my friends and acquaintances have ever wanted to see my passport, or asked to take my DNA to ascertain that I am
|
||||
a distinct human being from the other humans they know. Likewise, identifying me by my social connections is impractical
|
||||
as it would require an exceedingly weird amount of what can only be described as snooping. Yet, this concept of a
|
||||
single, consistent, global, true identity is exactly what up to now all technological solutions to the identity problem
|
||||
are trying to achieve.
|
||||
|
||||
Building Bridges
|
||||
================
|
||||
|
||||
I think I can offer you one main take-aways from the discussion above.
|
||||
|
||||
During modelling social systems, focus on relationships—not identity.
|
||||
|
||||
Rephrased into more actionable points, as someone designing a social digital system, do the following:
|
||||
|
||||
0. Early in the design stages, take the time to consider fundamental modelling issues like this one. If you don't, you
|
||||
will likely get stuck with a sub-optimal model that will be hard to get rid of.
|
||||
1. Where possible, be flexible. Allow people to chose their own identifier. Don't require them to use their real names,
|
||||
they may not wish to disclose those or they may not be in a format that is useful to you (they may be too long, too
|
||||
short, too ubiquituous, in foreign characters etc.). A free-form text field with a reasonable length limit is a good
|
||||
approach here.
|
||||
2. Do not use credit cards or phone numbers to identify people. There are many people who do not have either, and
|
||||
scammers can simply buy this data in bulk on the darknet.
|
||||
3. Allow people to create multiple identites [#accountswitchopsec]_, and acknowledge the role of social relationships in
|
||||
your interaction features. People have very legitimate reasons to separate areas of their lifes, and it is not for
|
||||
you or your computer to decide who is who to whom. If your thing requires a global search function, re-consider the
|
||||
data protection aspects of your system. If you want to encourage social functions in the face of bots and trolls,
|
||||
make it easy for people to share their identities out-of-band, such as through a QR code or a copy-and-pasteable
|
||||
short link. If you require someone's legal name or address for billing purposes, unify these identities behind the
|
||||
scenes if at all and allow them to act as if fully independent in public.
|
||||
|
||||
While change of perspective comes with its share of user experience challenges, but also with a promise for a more
|
||||
human, more dignified online experience. Perhaps we can find a way to adapt cyberspace to humans, instead of continuing
|
||||
trying it the other way around.
|
||||
|
||||
.. _astroturfing: https://en.wikipedia.org/wiki/Astroturfing
|
||||
.. _Stasi: https://en.wikipedia.org/wiki/Stasi
|
||||
|
||||
.. [#cryptocurrency] Pseudo-currencies in that, while they provide some aspects of a regular currency such as ownership
|
||||
and transactions, they lack most others. Traditional currencies are backed by states, regulated by central banks
|
||||
tasked with maintaining their stability and ultimately provide accountability through law enforcement, courts
|
||||
and political elections.
|
||||
|
||||
.. [#discriminatory] Discriminatory as in discriminating against minorities, but also as in deciding what is and what is
|
||||
not.
|
||||
|
||||
.. [#accountswitchopsec] This does mean that you should not actively prevent people from creating multiple accounts. It
|
||||
does not necessarily entail building a proper user interface around this practice. If you do the latter, e.g. by
|
||||
offering a "switch identity" button or an identiy drop-down menu on a post submission form, you can easily
|
||||
encourage slip-ups that might disclose the connection between two identities, and you make it possible for
|
||||
someone hacking a single login to learn about this connection as well.
|
||||
|
||||
.. [#meatspacefn] Meatspace_ is where people physically are, as opposed to cyberspace
|
||||
|
||||
.. _Meatspace: https://dictionary.cambridge.org/dictionary/english/meatspace
|
||||
.. _roles: https://en.wikipedia.org/wiki/Role
|
||||
.. _personhood: https://en.wikipedia.org/wiki/Personhood
|
||||
|
|
@ -1,89 +0,0 @@
|
|||
---
|
||||
title: "Identity between Cyberspace and Meatspace"
|
||||
date: 2020-09-09T15:00:00+02:00
|
||||
draft: true
|
||||
---
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure class="header">
|
||||
<img src="images/succulents.jpg">
|
||||
<figcaption>Photo by <a href="https://unsplash.com/@timbennettcreative">Tim Bennett</a> on <a href="https://unsplash.com/">Unsplash</a></figcaption>
|
||||
</figure>
|
||||
|
||||
Identity in Cyberspace
|
||||
======================
|
||||
|
||||
.. Identity is a frequent problem
|
||||
.. Easy solutions abound
|
||||
.. Precise modelling is uncommon
|
||||
.. True identity is sensitive, hard to handle
|
||||
..
|
||||
.. Often, conversational features emphasized -> true identity is unnecessary
|
||||
.. Social role theory
|
||||
.. Call to action
|
||||
|
||||
Most computer systems that interface with humans have a concept of user identity. The data structures used for its
|
||||
storage vary, but usually one *account* corresponds to one human *user*. In many applications, the system operator tries
|
||||
to ensure that one user cannot create multiple accounts. In online social networks, astrotufing_ and trolling are easier
|
||||
to fight when limits are imposed on account creation. In online stores, fraud prevention means the store operator needs
|
||||
their customers legal identity and the operator must be able to ban offending customers. In mobile messaging systems,
|
||||
users have to be able to find each other by some identifier such as name or phone number, and this identifier has to be
|
||||
unique and hard to forge.
|
||||
|
||||
Today, in systems that allow anyone to create an account have largely converged to require either an email address or a
|
||||
mobile phone number. Email addresses are used by systems that are less vulnerable to abuse and that are used on laptop
|
||||
or desktop computers. Mobile phone numbers are abundantly used in smartphone apps, as well as in systems more prone to
|
||||
abuse such as online social networks or ecommerce. Both are easily verified using a confirmation email or SMS.
|
||||
|
||||
When designing or programming an online system, it is uncommon that the precise real-world semantics of accounts are
|
||||
modelled. Most computer systems use ad-hoc data models. During their creation, their programmers implicit assumptions
|
||||
about the world are encoded into these data models. Most of the time this works fine, but it does lead to significant
|
||||
blind spots that can make systems break down for a fraction of their users.
|
||||
|
||||
Lives in Meatspace
|
||||
==================
|
||||
|
||||
A consequence of the proliferation of phone numbers being used to identify people is that most people will not be able
|
||||
to create multiple accounts. *"That's the point!"* you might say, but while we want to prevent scammers, spammers and
|
||||
boored schoolchildren from messing with our systems, everybody else may have legitimate reasons to have more than one
|
||||
account.
|
||||
|
||||
We can apply sociology's model of roles_ to understand this issue. In sociology, a role is the comprehensive pattern of
|
||||
rules and expectations that govern an individual's behavior corresponding to their social position. A key fact is that
|
||||
most people occupy mutliple roles. A parent may also be a company employee or a wife and perform accordingly given the
|
||||
circumstances. Systems that tie digital identity to legal personhood through the contracts behind phone numbers impede
|
||||
their users' attempts at role separation. Effects of this are e.g. that nowadays employers routinely screen applicants'
|
||||
social media accounts for unacceptable content.
|
||||
|
||||
While this role conflict merely amounts to a minor inconvenience to most there are many to who it poses an existential
|
||||
problem. Consider an LGBT+ person living in a repressive country or a politically conservative person living in a
|
||||
very liberal city. Both have legitimate reasons to strictly separate parts of their private lives from others. For both,
|
||||
much is at stake. Yet, both will have to practically circumvent most online systems registration barriers to implement
|
||||
this separation.
|
||||
|
||||
Trusting the User
|
||||
=================
|
||||
|
||||
While there is no single solution to these issues, there are several possible mitigations. The first and most important
|
||||
one is to systematically think about the system's data model when creating it. Which assumptions about the real world
|
||||
are inherent in it? Are these assumptions likely to cause issues? Ad-hoc models are easily created, but hard to get rid
|
||||
of when they start causing problems.
|
||||
|
||||
A general guideline on identity should be that hindering trolls by requiring things like phone numbers or credit card
|
||||
numbers is very likely to also be an obstacle to many entirely legitimate uses. Captchas_ or invitation links can help
|
||||
to keep out the trolls. Another approach is to limit the damage a troll can cause with things like effective moderation
|
||||
systems, reputation systems or by limiting the reach of newly created accounts.
|
||||
|
||||
Outside of e-commerce, actually tying a digital account to a real-world identity is very rarely necessary. The value of
|
||||
a messenger app is not in the names in its contacts list, but the conversations behind these names. When two people meet
|
||||
each other on the street, their interaction is shaped by a myriad of social factors—but *not* by them showing each other
|
||||
their photo ID.
|
||||
|
||||
Humans with their messy identities do not fit today's cyberspace well. Let's adapt cyberspace to humans, instead of
|
||||
trying it the other way around.
|
||||
|
||||
.. _astroturfing: https://en.wikipedia.org/wiki/Astroturfing
|
||||
.. _roles: https://en.wikipedia.org/wiki/Role
|
||||
.. _Captchas: https://link.springer.com/content/pdf/10.1007/3-540-39200-9_18.pdf
|
||||
|
||||
|
Before Width: | Height: | Size: 3.1 MiB |
|
Before Width: | Height: | Size: 1.7 MiB |
|
|
@ -1,60 +0,0 @@
|
|||
---
|
||||
title: "Thor's Hammer"
|
||||
date: 2018-05-03T11:59:37+02:00
|
||||
---
|
||||
|
||||
In case you were having an inferiority complex because your friends' IBM Model M keyboards are so much louder than the
|
||||
shitty rubber dome freebie you got with your pc... Here's the solution: Thor's Hammer, a simple typing cadence enhancer
|
||||
for `PS/2`_ keyboards.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<video controls loop>
|
||||
<source src="video/thors_hammer.mov" type="video/h264">
|
||||
<source src="video/thors_hammer.webm" type="video/webm">
|
||||
Your browser does not support the HTML5 video tag.
|
||||
</video>
|
||||
<figcaption>A demonstration of the completed project.
|
||||
|
||||
<a href="video/thors_hammer.mov">h264 download</a> /
|
||||
<a href="video/thors_hammer.webm">webm download</a>
|
||||
</figcaption>
|
||||
</figure>
|
||||
|
||||
The connects to the keyboard's PS/2 clock line and briefly actuates a large solenoid on each key press. An interesting
|
||||
fact about PS/2 is that the clock line is only active as long as either the host computer or the input device actually
|
||||
want to send data. In case of a keyboard that's the case when a key is pressed or when the host changes the keyboard's
|
||||
LED state, otherwise the clock line is silent. We ignore the LED activity for now as it's generally coupled to key
|
||||
presses. By just triggering an NE555 configured as astable flipflop we can stretch each train of clock pulses to a
|
||||
pulse a few tens of milliseconds long that is enough to actuate the solenoid.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/thors_hammer_schematic.jpg" alt="The schematic of the PS2 driver">
|
||||
<figcaption>The schematic of the driver stretching the PS/2 clock pulses to drive the solenoid.</figcaption>
|
||||
</figure>
|
||||
|
||||
|
||||
Since PS/2 sends each key press and key release separately this circuit will pulse twice per keystroke. It would be
|
||||
possible to ignore one of them but I figure the added noise just adds to the experience.
|
||||
|
||||
Built on a breadboard, the circuit looks like this.
|
||||
|
||||
.. raw:: html
|
||||
|
||||
<figure>
|
||||
<img src="images/thors_hammer_breadboard.jpg" alt="The circuit built on a breadboard">
|
||||
<figcaption>The completed circuit built up on a breadboard and attached to a keyboard.</figcaption>
|
||||
</figure>
|
||||
|
||||
|
||||
Since my solenoid did not have a tensioning spring I used a rubber band and some vinyl tape to make an adjustable
|
||||
tensioner. The small orange USB hub serves as an end-stop because I had nothing else of the right shape. The sound and
|
||||
resonance of the thing can be adjusted to taste by moving the end stop, adjusting the tensioning rubber and tuning the
|
||||
excitation duration using the potentiometer. My particular solenoid was a bit slow so I added some pieces of circuit
|
||||
board as shims between the plunger and the case to limit the plunger's travel inside the solenoid core.
|
||||
|
||||
.. _`PS/2`: https://en.wikipedia.org/wiki/PS/2_port
|
||||
|
||||
|
Before Width: | Height: | Size: 111 KiB |
|
Before Width: | Height: | Size: 2.8 MiB |
|
Before Width: | Height: | Size: 799 KiB |
|
Before Width: | Height: | Size: 3.5 MiB |