-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathIMX6ULL嵌入式Linux驱动学习笔记(十).html
42 lines (40 loc) · 332 KB
/
IMX6ULL嵌入式Linux驱动学习笔记(十).html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
<!DOCTYPE html><html><head><meta charset="utf-8"><meta name="renderer" content="webkit"><meta http-equiv="X-UA-Compatible" content="IE=edge"><link rel="dns-prefetch" href="https://proudrabbit.gitee.io"><title>IMX6ULL嵌入式Linux驱动学习笔记(十) | 兔子的个人博客 - Hexo Blog</title><meta name="generator" content="hexo-theme-yilia-plus"><meta name="viewport" content="width=device-width,initial-scale=1,maximum-scale=1"><meta name="description" content="正点原子alpha开发板IMX6ULL嵌入式Linux开发学习笔记。"><meta property="og:type" content="article"><meta property="og:title" content="IMX6ULL嵌入式Linux驱动学习笔记(十)"><meta property="og:url" content="https://proudrabbit.gitee.io/IMX6ULL%E5%B5%8C%E5%85%A5%E5%BC%8FLinux%E9%A9%B1%E5%8A%A8%E5%AD%A6%E4%B9%A0%E7%AC%94%E8%AE%B0%EF%BC%88%E5%8D%81%EF%BC%89.html"><meta property="og:site_name" content="兔子的个人博客 - Hexo Blog"><meta property="og:description" content="正点原子alpha开发板IMX6ULL嵌入式Linux开发学习笔记。"><meta property="og:locale" content="zh_CN"><meta property="og:image" content="https://i.loli.net/2020/12/23/Hb3IhCd95umSLcD.png"><meta property="og:image" content="https://i.loli.net/2020/12/23/TPzJBZXh7eqCg6V.png"><meta property="article:published_time" content="2021-01-08T03:21:09.000Z"><meta property="article:modified_time" content="2021-08-25T14:11:38.102Z"><meta property="article:author" content="路痴的兔子"><meta property="article:tag" content="嵌入式"><meta property="article:tag" content="IMX6ULL"><meta property="article:tag" content="学习笔记"><meta name="twitter:card" content="summary"><meta name="twitter:image" content="https://i.loli.net/2020/12/23/Hb3IhCd95umSLcD.png"><link rel="alternative" href="/atom.xml" title="兔子的个人博客 - Hexo Blog" type="application/atom+xml"><link rel="icon" href="/img/favicon.ico"><link rel="apple-touch-icon" href="/apple-touch-icon-180x180.png"><link rel="stylesheet" type="text/css" href="/./main.b8fa34.css"><style type="text/css">#container.show{background:linear-gradient(200deg,#a0cfe4,#e8c37e)}</style><script src="/lib/clickLove.js"></script><script>!function(){var t=document.createElement("script"),e=window.location.protocol.split(":")[0];t.src="https"===e?"https://zz.bdstatic.com/linksubmit/push.js":"http://push.zhanzhang.baidu.com/push.js";var s=document.getElementsByTagName("script")[0];s.parentNode.insertBefore(t,s)}()</script><script src="/lib/pace.min.js"></script><link href="//cdn.bootcss.com/pace/1.0.2/themes/silver/pace-theme-corner-indicator.css" rel="stylesheet"><meta name="google-site-verification" content="AJuUAh5QvaKZUGnT5rGD-WA_VxSxRy1s-eQgGDhiHrY"><meta name="msvalidate.01" content="69AE9CE05F5D0FD01E37DE9CF11E1E6B"><meta name="360-site-verification" content="d52524f63d68a2e4be5584bc63a0dd44"><link rel="stylesheet" href="\assets\css\APlayer.min.css" class="aplayer-style-marker">
<script src="\assets\js\APlayer.min.js" class="aplayer-script-marker"></script>
<script src="\assets\js\Meting.min.js" class="meting-script-marker"></script>
</head><body><div id="container" q-class="show:isCtnShow"><canvas id="anm-canvas" class="anm-canvas"></canvas><div class="left-col" q-class="show:isShow"><div class="overlay" style="background:rgba(77,77,77,.7)"></div><div class="intrude-less"><header id="header" class="inner"><a href="/" class="profilepic"><img src="/img/head2.jpg" class="js-avatar"></a><hgroup><h1 class="header-author"><a href="/">路痴的兔子</a></h1></hgroup><p class="header-subtitle">非学无以广才,非志无以成学</p><nav class="header-menu"><ul><li><a href="https://proudrabbit.gitee.io/resume" target="_blank">个人简历</a></li><li><a href="https://proudrabbit.gitee.io/love_story" target="_blank">Love Story</a></li><li><a href="/tags/%E5%B5%8C%E5%85%A5%E5%BC%8F/" target="_blank">嵌入式</a></li><li><a href="/tags/Linux/" target="_blank">Linux系统</a></li><li><a href="/tags/%E7%BC%96%E7%A8%8B%E8%AF%AD%E8%A8%80/" target="_blank">编程语言</a></li><li><a href="/tags/%E8%BD%AF%E4%BB%B6%E4%BD%BF%E7%94%A8/" target="_blank">软件使用</a></li><li><a href="/tags/%E9%9A%8F%E7%AC%94/" target="_blank">随笔</a></li><li><a href="/archives/index.html" target="_blank">归档</a></li></ul></nav><nav class="header-smart-menu"><a q-on="click: openSlider(e, 'innerArchive')" href="javascript:void(0)">所有文章</a> <a q-on="click: openSlider(e, 'friends')" href="javascript:void(0)">友情链接</a> <a q-on="click: openSlider(e, 'aboutme')" href="javascript:void(0)">关于我</a></nav><nav class="header-nav"><div class="social"><a class="github" href="/~https://github.com/ProudRabbit" title="GitHub" target="_blank"><i class="icon-github"></i></a> <a class="gitee" href="https://gitee.com/proudrabbit" title="码云" target="_blank"><i class="icon-gitee"></i></a> <a class="csdn" href="https://blog.csdn.net/qq_37284607" title="CSDN" target="_blank"><i class="icon-csdn"></i></a> <a class="bilibili" href="https://space.bilibili.com/15573126" title="哔哩哔哩" target="_blank"><i class="icon-bilibili"></i></a> <a class="qq" href="/img/qrcode.png" title="QQ" target="_blank"><i class="icon-qq"></i></a> <a class="mail" href="mailto:yuh_feng@163.com" title="NULL" target="_blank"><i class="icon-mail"></i></a></div><p style="font-size:12px">听首音乐放松一下吧o(* ̄▽ ̄*)o</p><p></p><div id="aplayer-hjnObCgk" class="aplayer aplayer-tag-marker meting-tag-marker" data-id="8271608824" data-server="tencent" data-type="playlist" data-loop="none" data-order="random" data-lrctype="0" data-listfolded="true" data-fixed="true" data-autoplay="false" data-volume="0.4" data-mutex="true" data-listmaxheight="400px" data-preload="none" data-theme=""></div></nav></header></div></div><div class="mid-col" q-class="show:isShow,hide:isShow|isFalse"><a class="forkMe" style="position:absolute;z-index:999;top:0;right:.5em" href="/~https://github.com/ProudRabbit" target="_blank"><img src="/img/forkme.png" class="attachment-full size-full" alt="Fork me on GitHub" data-recalc-dims="1"></a><nav id="mobile-nav"><div class="overlay js-overlay" style="background:rgba(77,77,77,.7)"></div><div class="btnctn js-mobile-btnctn"><div class="slider-trigger list" q-on="click: openSlider(e)"><i class="icon icon-sort"></i></div></div><div class="intrude-less"><header id="header" class="inner"><div class="profilepic"><a href="/"><img src="/img/head2.jpg" class="js-avatar"></a></div><hgroup><h1 class="header-author js-header-author">路痴的兔子</h1></hgroup><p class="header-subtitle"><i class="icon icon-quo-left"></i>非学无以广才,非志无以成学<i class="icon icon-quo-right"></i></p><nav class="header-nav"><div class="social"><a class="github" target="_blank" href="/~https://github.com/ProudRabbit" title="GitHub"><i class="icon-github"></i></a> <a class="gitee" target="_blank" href="https://gitee.com/proudrabbit" title="码云"><i class="icon-gitee"></i></a> <a class="csdn" target="_blank" href="https://blog.csdn.net/qq_37284607" title="CSDN"><i class="icon-csdn"></i></a> <a class="bilibili" target="_blank" href="https://space.bilibili.com/15573126" title="哔哩哔哩"><i class="icon-bilibili"></i></a> <a class="qq" target="_blank" href="/img/qrcode.png" title="QQ"><i class="icon-qq"></i></a> <a class="mail" target="_blank" href="mailto:yuh_feng@163.com" title="NULL"><i class="icon-mail"></i></a></div></nav><nav class="header-menu js-header-menu"><ul style="width:70%"><li style="width:12.5%"><a href="https://proudrabbit.gitee.io/resume">个人简历</a></li><li style="width:12.5%"><a href="https://proudrabbit.gitee.io/love_story">Love Story</a></li><li style="width:12.5%"><a href="/tags/%E5%B5%8C%E5%85%A5%E5%BC%8F/">嵌入式</a></li><li style="width:12.5%"><a href="/tags/Linux/">Linux系统</a></li><li style="width:12.5%"><a href="/tags/%E7%BC%96%E7%A8%8B%E8%AF%AD%E8%A8%80/">编程语言</a></li><li style="width:12.5%"><a href="/tags/%E8%BD%AF%E4%BB%B6%E4%BD%BF%E7%94%A8/">软件使用</a></li><li style="width:12.5%"><a href="/tags/%E9%9A%8F%E7%AC%94/">随笔</a></li><li style="width:12.5%"><a href="/archives/index.html">归档</a></li></ul></nav></header></div><div class="mobile-mask" style="display:none" q-show="isShow"></div></nav><div id="wrapper" class="body-wrap"><div class="menu-l"><div class="canvas-wrap"><canvas data-colors="#eaeaea" data-sectionheight="100" data-contentid="js-content" id="myCanvas1" class="anm-canvas"></canvas></div><div id="js-content" class="content-ll"><article id="post-IMX6ULL嵌入式Linux驱动学习笔记(十)" class="article article-type-post" itemscope itemprop="blogPost"><div class="article-inner"><header class="article-header"><h1 class="article-title" itemprop="name">IMX6ULL嵌入式Linux驱动学习笔记(十)</h1><span id="busuanzi_container_page_pv" style="display:none" class="archive-article-date"><i class="icon-smile icon"></i> 阅读数:<span id="busuanzi_value_page_pv"></span>次 </span><a href="/IMX6ULL%E5%B5%8C%E5%85%A5%E5%BC%8FLinux%E9%A9%B1%E5%8A%A8%E5%AD%A6%E4%B9%A0%E7%AC%94%E8%AE%B0%EF%BC%88%E5%8D%81%EF%BC%89.html" class="archive-article-date"><time datetime="2021-01-08T03:21:09.000Z" itemprop="datePublished"><i class="icon-calendar icon"></i>2021-01-08</time></a><div style="margin-top:10px"><span class="post-time"><span class="post-meta-item-icon"><i class="icon-statistics"></i> <span class="post-meta-item-text">字数统计:</span> <span class="post-count">8.5k字</span> </span></span><span class="post-time"> | <span class="post-meta-item-icon"><i class="icon-book icon"></i> <span class="post-meta-item-text">阅读时长≈</span> <span class="post-count">40分</span></span></span></div></header><div class="article-entry" itemprop="articleBody"><p><strong>IMX6ULL嵌入式Linux驱动开发学习</strong></p><p>以下内容是我在学习正点原子<code>IMX6ULL</code>开发板<code>alpha</code>中记录的笔记,部分摘录自正点原子<code>IMX6ULL开发手册</code>。</p><h2 id="一、Linux阻塞和非阻塞IO"><a href="#一、Linux阻塞和非阻塞IO" class="headerlink" title="一、Linux阻塞和非阻塞IO"></a>一、Linux阻塞和非阻塞IO</h2><h3 id="1-1-阻塞和非阻塞简介"><a href="#1-1-阻塞和非阻塞简介" class="headerlink" title="1.1 阻塞和非阻塞简介"></a>1.1 阻塞和非阻塞简介</h3><p>这里的 <code>IO</code> 指的是 <code>Input/Output</code>,也就是输入/输出,是应用程序对驱动设备的输入/输出操作。当应用程序对设备驱动进行操作的时候,如果不能获取到设备资源,那么阻塞式 <code>IO</code> 就会将应用程序对应的线程挂起,直到设备资源可以获取为止。对于非阻塞 <code>IO</code>,应用程序对应的线程不会挂起,它要么一直轮询等待,直到设备资源可以使用,要么就直接放弃。</p><a id="more"></a><ul><li><p>阻塞式<code>IO</code>:</p><blockquote><p>当资源不可用时,应用程序就会挂起。当资源可用的时候,唤醒任务。应用程序使用<code>open</code>打开驱动文件,默认是阻塞方式打开。</p><p>阻塞式<code>IO</code>访问示意图:</p><p><img src="https://i.loli.net/2020/12/23/Hb3IhCd95umSLcD.png" alt="阻塞式IO" title="阻塞式IO"></p></blockquote></li><li><p>非阻塞式<code>IO</code>:</p><blockquote><p>当资源不可用的时候,应用程序轮询查看,或放弃。会有超时处理。应用程序如果想在使用<code>open</code>打开驱动文件时使用非阻塞的方式打开,需要使用<code>O_NONBLOCK</code>。</p><p>非阻塞式<code>IO</code>访问示意图:</p><p><img src="https://i.loli.net/2020/12/23/TPzJBZXh7eqCg6V.png" alt="非阻塞式IO" title="非阻塞式IO"></p></blockquote></li></ul><h3 id="1-2-等待队列(阻塞访问)"><a href="#1-2-等待队列(阻塞访问)" class="headerlink" title="1.2 等待队列(阻塞访问)"></a>1.2 等待队列(阻塞访问)</h3><ol><li><p>等待队列头</p><p>阻塞访问最大的好处就是当设备文件不可操作的时候进程可以进入休眠态,这样可以将<code>CPU</code> 资源让出来。但是,当设备文件可以操作的时候就必须唤醒进程,一般在中断函数里面完成唤醒工作。 <code>Linux</code>内核提供了等待队列(<code>wait queue</code>)来实现阻塞进程的唤醒工作,如果我们要在驱动中使用等待队列,必须创建并初始化一个等待队列头,等待队列头使用结构体<code>wait_queue_head_t</code> 表示:</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br></pre></td><td class="code"><pre><span class="line"><span class="class"><span class="keyword">struct</span> __<span class="title">wait_queue_head</span> {</span></span><br><span class="line"> <span class="keyword">spinlock_t</span> lock;</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">list_head</span> <span class="title">task_list</span>;</span></span><br><span class="line">};</span><br><span class="line"><span class="keyword">typedef</span> <span class="class"><span class="keyword">struct</span> __<span class="title">wait_queue_head</span> <span class="title">wait_queue_head_t</span>;</span></span><br></pre></td></tr></table></figure><p>定义好等待队列头以后需要初始化,使用 <code>init_waitqueue_head</code> 函数初始化等待队列头,函数原型如下:</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @param q 等待队列头</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">void</span> <span class="title">init_waitqueue_head</span><span class="params">(<span class="keyword">wait_queue_head_t</span> *q)</span></span></span><br></pre></td></tr></table></figure><p>也可以使用宏 <code>DECLARE_WAIT_QUEUE_HEAD</code> 来一次性完成等待队列头的定义和初始化。</p></li><li><p>等待队列项</p><p>等待队列头就是一个等待队列的头部,每个访问设备的进程都是一个队列项,当设备不可用的时候就要将这些进程对应的等待队列项添加到等待队列里面。结构体 <code>wait_queue_t</code> 表示等待队列项。</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line"><span class="class"><span class="keyword">struct</span> __<span class="title">wait_queue</span> {</span></span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">int</span> flags;</span><br><span class="line"> <span class="keyword">void</span> *<span class="keyword">private</span>;</span><br><span class="line"> <span class="keyword">wait_queue_func_t</span> func;</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">list_head</span> <span class="title">task_list</span>;</span></span><br><span class="line">};</span><br><span class="line"><span class="keyword">typedef</span> <span class="class"><span class="keyword">struct</span> __<span class="title">wait_queue</span> <span class="title">wait_queue_t</span>;</span></span><br></pre></td></tr></table></figure><p>同样可以使用宏 <code>DECLARE_WAITQUEUE</code> 定义并初始化一个等待队列项。</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br></pre></td><td class="code"><pre><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @param name 等待队列项的名字</span></span><br><span class="line"><span class="comment"> * @param tsk 表示该等待队列项属于哪个任务(进程),一般为 current</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line">DECLARE_WAITQUEUE(name, tsk)</span><br></pre></td></tr></table></figure></li><li><p>添加/移除等待队列项</p><p>当设备不可访问的时候就需要将进程对应的等待队列项添加到前面创建的等待队列头中,只有添加到等待队列头中以后进程才能进入休眠态。当设备可以访问以后再将进程对应的等待队列项从等待队列头中移除即可。</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br></pre></td><td class="code"><pre><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 添加等待队列项到等待队列头</span></span><br><span class="line"><span class="comment"> * @param q 等待队列头</span></span><br><span class="line"><span class="comment"> * @param wait 等待队列项</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">void</span> <span class="title">add_wait_queue</span><span class="params">(<span class="keyword">wait_queue_head_t</span> *q,<span class="keyword">wait_queue_t</span> *wait)</span></span>;</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 将等待队列项从等待队列头中移除</span></span><br><span class="line"><span class="comment"> * @param q 等待队列头</span></span><br><span class="line"><span class="comment"> * @param wait 等待队列项</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">void</span> <span class="title">remove_wait_queue</span><span class="params">(<span class="keyword">wait_queue_head_t</span> *q,<span class="keyword">wait_queue_t</span> *wait)</span></span></span><br></pre></td></tr></table></figure></li><li><p>等待唤醒</p><p>当设备可以使用的时候就要唤醒进入休眠态的进程,唤醒可以使用如下两个函数。</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br></pre></td><td class="code"><pre><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 唤醒等待队列头下所有的进程</span></span><br><span class="line"><span class="comment"> * @param q 要唤醒的等待队列头</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">void</span> <span class="title">wake_up</span><span class="params">(<span class="keyword">wait_queue_head_t</span> *q)</span></span>;</span><br><span class="line"><span class="function"><span class="keyword">void</span> <span class="title">wake_up_interruptible</span><span class="params">(<span class="keyword">wait_queue_head_t</span> *q)</span></span>;</span><br></pre></td></tr></table></figure><p><strong><code>wake_up</code> 函数可以唤醒处于 <code>TASK_INTERRUPTIBLE</code> 和 <code>TASK_UNINTERRUPTIBLE</code> 状态的进程,而 <code>wake_up_interruptible</code> 函数只能唤醒处于 <code>TASK_INTERRUPTIBLE</code> 状态的进程。</strong></p></li><li><p>等待事件</p><p>除了主动唤醒以外,也可以设置等待队列等待某个事件,当这个事件满足以后就自动唤醒等待队列中的进程,和等待事件有关的<code>API</code>函数如下。</p><table><thead><tr><th>函数</th><th>描述</th></tr></thead><tbody><tr><td>wait_event(wq, condition)</td><td>等待以 <code>wq</code> 为等待队列头的等待队列被唤醒,前提是 <code>condition</code> 条件必须满足(为真),否则一直阻塞。此 函 数 会 将 进 程 设 置 为<code>TASK_UNINTERRUPTIBLE</code> 状态。</td></tr><tr><td>wait_event_timeout(wq, condition, timeout)</td><td>功能和 <code>wait_event</code> 类似,但是此函数可以添加超时时间,以 <code>jiffies</code> 为单位。此函数有返回值,如果返回 <code>0</code> 的话表示超时时间到,而且 <code>condition</code> 为假。为 <code>1</code> 的话表示 <code>condition</code> 为真,也就是条件满足了。</td></tr><tr><td>wait_event_interruptible(wq, condition)</td><td>与 <code>wait_event</code> 函数类似,但是此函数将进程设置为<code>TASK_INTERRUPTIBLE</code>,就是可以被信号打断。</td></tr><tr><td>wait_event_interruptible_timeout(wq,condition, timeout)</td><td>与 <code>wait_event_timeout</code> 函数类似,此函数也将进 程设置为<code>TASK_INTERRUPTIBLE</code>,可以被信号打断。</td></tr></tbody></table></li></ol><h3 id="1-3-轮询(非阻塞访问)"><a href="#1-3-轮询(非阻塞访问)" class="headerlink" title="1.3 轮询(非阻塞访问)"></a>1.3 轮询(非阻塞访问)</h3><p>如果用户应用程序以非阻塞的方式访问设备,设备驱动程序就要提供非阻塞的处理方式,也就是轮询。 <code>poll</code>、 <code>epoll</code> 和 <code>select</code> 可以用于处理轮询,应用程序通过 <code>select</code>、 <code>epoll</code> 或 <code>poll</code> 函数来查询设备是否可以操作,如果可以操作的话就从设备读取或者向设备写入数据。</p><p>当应用程序调用 <code>select</code>、 <code>epoll</code> 或 <code>poll</code> 函数的时候,设备驱动程序中的 <code>poll</code> 函数就会执行,因此需要在设备驱动程序中编写 <code>poll</code> 函数。</p><p>驱动里<code>poll</code>函数原型如下。</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br></pre></td><td class="code"><pre><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @param filp 要打开的设备文件(文件描述符)</span></span><br><span class="line"><span class="comment"> * @param wait 结构体 poll_table_struct 类型指针,由应用程序传递进来。一般将此参数传递给 poll_wait 函数。</span></span><br><span class="line"><span class="comment"> * @return 向应用程序返回设备或者资源状态。</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">unsigned</span> <span class="title">int</span> <span class="params">(*poll)</span><span class="params">(struct file *filp, struct poll_table_struct *wait)</span></span>;</span><br></pre></td></tr></table></figure><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br></pre></td><td class="code"><pre><span class="line"><span class="comment">/* return 可以返回的资源状态 */</span></span><br><span class="line">POLLIN <span class="comment">// 有数据可以读取。</span></span><br><span class="line">POLLPRI <span class="comment">// 有紧急的数据需要读取。</span></span><br><span class="line">POLLOUT <span class="comment">// 可以写数据。</span></span><br><span class="line">POLLERR <span class="comment">// 指定的文件描述符发生错误。</span></span><br><span class="line">POLLHUP <span class="comment">// 指定的文件描述符挂起。</span></span><br><span class="line">POLLNVAL <span class="comment">// 无效的请求。</span></span><br><span class="line">POLLRDNORM <span class="comment">// 等同于 POLLIN,普通数据可读</span></span><br></pre></td></tr></table></figure><p>需要在驱动程序的 <code>poll</code> 函数中调用 <code>poll_wait</code> 函数, <code>poll_wait</code> 函数不会引起阻塞,只是将应用程序添加到 <code>poll_table</code> 中。</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br></pre></td><td class="code"><pre><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @param filp 文件描述符</span></span><br><span class="line"><span class="comment"> * @param wait_address 要添加到 poll_table 中的等待队列头</span></span><br><span class="line"><span class="comment"> * @param p poll_table,就是file_operations 中 poll 函数的 wait 参数</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">void</span> <span class="title">poll_wait</span><span class="params">(struct file *filp, <span class="keyword">wait_queue_head_t</span> *wait_address, poll_table *p)</span></span></span><br></pre></td></tr></table></figure><ol><li><p><code>select</code>函数</p><blockquote><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br></pre></td><td class="code"><pre><span class="line">><span class="comment">/**</span></span><br><span class="line"><span class="comment">>* @param nfds 所要监视的这三类文件描述集合中,最大文件描述符加1。</span></span><br><span class="line"><span class="comment">>* @param readfds 监视指定描述符集的读变化,也就是监视文件是否可读</span></span><br><span class="line"><span class="comment">>* @param writefds 监视文件是否可以进行写操作</span></span><br><span class="line"><span class="comment">>* @param exceptfds 监视这些文件的异常</span></span><br><span class="line"><span class="comment">>* @param timeout 超时时间,为 NULL 的时候就表示无限期的等待。</span></span><br><span class="line"><span class="comment">>* @return 0,表示的话就表示超时发生,但是没有任何文件描述符可以进行操作;-1,发生错误;其他值,可以进行操作的文件描述符个数。</span></span><br><span class="line"><span class="comment">>*/</span></span><br><span class="line">><span class="function"><span class="keyword">int</span> <span class="title">select</span><span class="params">(<span class="keyword">int</span> nfds, fd_set *readfds, fd_set *writefds, </span></span></span><br><span class="line"><span class="function"><span class="params"> fd_set *exceptfds, struct timeval *timeout)</span></span>;</span><br><span class="line"></span><br><span class="line">><span class="comment">/* timeval结构体 */</span></span><br><span class="line">><span class="class"><span class="keyword">struct</span> <span class="title">timeval</span> {</span></span><br><span class="line"><span class="keyword">long</span> tv_sec; <span class="comment">/* 秒 */</span></span><br><span class="line"><span class="keyword">long</span> tv_usec; <span class="comment">/* 微妙 */</span></span><br><span class="line">>};</span><br></pre></td></tr></table></figure><p>比如现在要从一个设备文件中读取数据,那么就可以定义一个 <code>fd_set</code> 变量,这个变量要传递给参数 <code>readfds</code>。当我们定义好一个 <code>fd_set</code> 变量以后可以使用如下所示几个宏进行操作</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line">><span class="function"><span class="keyword">void</span> <span class="title">FD_ZERO</span><span class="params">(fd_set *<span class="built_in">set</span>)</span></span>;</span><br><span class="line">><span class="function"><span class="keyword">void</span> <span class="title">FD_SET</span><span class="params">(<span class="keyword">int</span> fd, fd_set *<span class="built_in">set</span>)</span></span>;</span><br><span class="line">><span class="function"><span class="keyword">void</span> <span class="title">FD_CLR</span><span class="params">(<span class="keyword">int</span> fd, fd_set *<span class="built_in">set</span>)</span></span>;</span><br><span class="line">><span class="function"><span class="keyword">int</span> <span class="title">FD_ISSET</span><span class="params">(<span class="keyword">int</span> fd, fd_set *<span class="built_in">set</span>)</span></span>;</span><br></pre></td></tr></table></figure><p><code>FD_ZERO</code> 用于将 <code>fd_set</code> 变量的所有位都清零, <code>FD_SET</code> 用于将 <code>fd_set</code> 变量的某个位置 <code>1</code>,也就是向 <code>fd_set</code> 添加一个文件描述符,参数 <code>fd</code> 就是要加入的文件描述符。<code>FD_CLR</code> 用于将<code>fd_set</code>变量的某个位清零,也就是将一个文件描述符从 <code>fd_set</code>中删除,参数 <code>fd</code> 就是要删除的文件描述符。 <code>FD_ISSET</code> 用于测试一个文件是否属于某个集合,参数 <code>fd</code> 就是要判断的文件描述符。</p><p>使用 <code>select</code> 函数对某个设备驱动文件进行读非阻塞访问的操作示例如下</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br></pre></td><td class="code"><pre><span class="line">><span class="function"><span class="keyword">void</span> <span class="title">main</span><span class="params">(<span class="keyword">void</span>)</span></span></span><br><span class="line"><span class="function">></span>{</span><br><span class="line"><span class="keyword">int</span> ret, fd; <span class="comment">/* 要监视的文件描述符 */</span></span><br><span class="line">fd_set readfds; <span class="comment">/* 读操作文件描述符集 */</span></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">timeval</span> <span class="title">timeout</span>;</span> <span class="comment">/* 超时结构体 */</span></span><br><span class="line"></span><br><span class="line">fd = <span class="built_in">open</span>(<span class="string">"dev_xxx"</span>, O_RDWR | O_NONBLOCK); <span class="comment">/* 非阻塞式访问 */</span></span><br><span class="line"></span><br><span class="line">FD_ZERO(&readfds); <span class="comment">/* 清除 readfds */</span></span><br><span class="line">FD_SET(fd, &readfds); <span class="comment">/* 将 fd 添加到 readfds 里面 */</span></span><br><span class="line"></span><br><span class="line"><span class="comment">/* 构造超时时间 */</span></span><br><span class="line">timeout.tv_sec = <span class="number">0</span>;</span><br><span class="line">timeout.tv_usec = <span class="number">500000</span>; <span class="comment">/* 500ms */</span></span><br><span class="line"></span><br><span class="line">ret = select(fd + <span class="number">1</span>, &readfds, <span class="literal">NULL</span>, <span class="literal">NULL</span>, &timeout);</span><br><span class="line"><span class="keyword">switch</span> (ret) {</span><br><span class="line"> <span class="keyword">case</span> <span class="number">0</span>: <span class="comment">/* 超时 */</span></span><br><span class="line"> <span class="built_in">printf</span>(<span class="string">"timeout!\r\n"</span>);</span><br><span class="line"> <span class="keyword">break</span>;</span><br><span class="line"> <span class="keyword">case</span> <span class="number">-1</span>: <span class="comment">/* 错误 */</span></span><br><span class="line"> <span class="built_in">printf</span>(<span class="string">"error!\r\n"</span>);</span><br><span class="line"> <span class="keyword">break</span>;</span><br><span class="line"> <span class="keyword">default</span>: <span class="comment">/* 可以读取数据 */</span></span><br><span class="line"> <span class="keyword">if</span>(FD_ISSET(fd, &readfds)) { <span class="comment">/* 判断是否为 fd 文件描述符 */</span></span><br><span class="line"> <span class="comment">/* 使用 read 函数读取数据 */</span></span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">break</span>;</span><br><span class="line">}</span><br><span class="line">>}</span><br></pre></td></tr></table></figure></blockquote></li><li><p><code>poll</code>函数</p><blockquote><p>在单个线程中, <code>select</code> 函数能够监视的文件描述符数量有最大的限制,一般为 <code>1024</code>,可以修改内核将监视的文件描述符数量改大,但是这样会降低效率!这个时候就可以使用 <code>poll</code> 函数, <code>poll</code> 函数本质上和 <code>select</code> 没有太大的差别,但是 <code>poll</code> 函数没有最大文件描述符限制。</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br></pre></td><td class="code"><pre><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @param fds 要监视的文件描述符集合以及要监视的事件,为一个数组</span></span><br><span class="line"><span class="comment"> * @param nfds poll函数要监视的文件描述符数量</span></span><br><span class="line"><span class="comment"> * @param timeout 超时时间,单位为 ms。</span></span><br><span class="line"><span class="comment"> * @return 返回 revents 域中不为 0 的 pollfd 结构体个数,也就是发生事件或错误的文件描述符数量;0,超时;-1,发生错误,并且设置 errno 为错误类型</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">int</span> <span class="title">poll</span><span class="params">(struct pollfd *fds, <span class="keyword">nfds_t</span> nfds, <span class="keyword">int</span> timeout)</span></span>;</span><br><span class="line"></span><br><span class="line"><span class="comment">/* pollfd 结构体 */</span></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">pollfd</span> {</span></span><br><span class="line"> <span class="keyword">int</span> fd; <span class="comment">/* 文件描述符 */</span></span><br><span class="line"> short events; <span class="comment">/* 请求的事件 */</span></span><br><span class="line"> short revents; <span class="comment">/* 返回的事件 */</span></span><br><span class="line">};</span><br></pre></td></tr></table></figure><p><code>pollfd</code> 结构体中 <code>fd</code> 是要监视的文件描述符,如果 <code>fd</code> 无效的话那么 <code>events</code> 监视事件也就无效,并且 <code>revents</code>返回 <code>0</code>。 <code>events</code>是要监视的事件,可监视的事件类型如下所示, <code>revents</code> 是返回参数,也就是返回的事件, 由 <code>Linux</code> 内核设置具体的返回事件</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line">POLLIN <span class="comment">/* 有数据可以读取。*/</span></span><br><span class="line">POLLPRI <span class="comment">/* 有紧急的数据需要读取。 */</span></span><br><span class="line">POLLOUT <span class="comment">/* 可以写数据。 */</span></span><br><span class="line">POLLERR <span class="comment">/* 指定的文件描述符发生错误。 */</span></span><br><span class="line">POLLHUP <span class="comment">/* 指定的文件描述符挂起。 */</span></span><br><span class="line">POLLNVAL <span class="comment">/* 无效的请求。 */</span></span><br><span class="line">POLLRDNORM <span class="comment">/* 等同于 POLLIN */</span></span><br></pre></td></tr></table></figure><p>使用 <code>poll</code> 函数对某个设备驱动文件进行读非阻塞访问的操作示例如下</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">void</span> <span class="title">main</span><span class="params">(<span class="keyword">void</span>)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> ret;</span><br><span class="line"> <span class="keyword">int</span> fd; <span class="comment">/* 要监视的文件描述符 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">pollfd</span> <span class="title">fds</span>;</span></span><br><span class="line"></span><br><span class="line"> fd = <span class="built_in">open</span>(filename, O_RDWR | O_NONBLOCK); <span class="comment">/* 非阻塞式访问 */</span></span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 构造结构体 */</span></span><br><span class="line"> fds.fd = fd;</span><br><span class="line"> fds.events = POLLIN; <span class="comment">/* 监视数据是否可以读取 */</span></span><br><span class="line"></span><br><span class="line"> ret = poll(&fds, <span class="number">1</span>, <span class="number">500</span>); <span class="comment">/* 轮询文件是否可操作,超时 500ms */</span></span><br><span class="line"> <span class="keyword">if</span> (ret) { <span class="comment">/* 数据有效 */</span></span><br><span class="line"> ......</span><br><span class="line"> <span class="comment">/* 读取数据 */</span></span><br><span class="line"> ......</span><br><span class="line"> } <span class="keyword">else</span> <span class="keyword">if</span> (ret == <span class="number">0</span>) { <span class="comment">/* 超时 */</span></span><br><span class="line"> ......</span><br><span class="line"></span><br><span class="line"> } <span class="keyword">else</span> <span class="keyword">if</span> (ret < <span class="number">0</span>) { <span class="comment">/* 错误 */</span></span><br><span class="line"> ......</span><br><span class="line"></span><br><span class="line"> }</span><br><span class="line">}</span><br></pre></td></tr></table></figure></blockquote></li><li><p><code>epoll</code>函数</p><blockquote><p>传统的 <code>selcet</code> 和 <code>poll</code> 函数都会随着所监听的 <code>fd</code> 数量的增加,出现效率低下的问题,而且 <code>poll</code> 函数每次必须遍历所有的描述符来检查就绪的描述符,这个过程很浪费时间。为此, <code>epoll</code> 应运而生, <code>epoll</code> 就是为处理大并发而准备的,一般常常在网络编程中使用 <code>epoll</code> 函数。应用程序需要先使用 <code>epoll_create</code> 函数创建一个 <code>epoll</code> 句柄。</p><figure class="highlight"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br></pre></td><td class="code"><pre><span class="line"> ><span class="comment">/**</span></span><br><span class="line"><span class="comment">* @param size 从Linux2.6.8开始此参数已无意义,随便填写一个大于0的值即可。</span></span><br><span class="line"><span class="comment">* @return epoll句柄,如果为-1的话表示创建失败。</span></span><br><span class="line"><span class="comment">*/</span></span><br><span class="line"> >int epoll_create(int size);</span><br></pre></td></tr></table></figure><p><code>epoll</code> 句柄创建成功以后使用 <code>epoll_ctl</code> 函数向其中添加要监视的文件描述符以及监视的事件。</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br></pre></td><td class="code"><pre><span class="line"> ><span class="comment">/**</span></span><br><span class="line"><span class="comment">* @param epfd 要操作的epoll句柄,也就是使用epoll_create函数创建的epoll句柄。</span></span><br><span class="line"><span class="comment">* @param op 表示要对 epfd(epoll 句柄)进行的操作。</span></span><br><span class="line"><span class="comment">* @param fd 要监视的文件描述符。</span></span><br><span class="line"><span class="comment">* @param event 要监视的事件类型。</span></span><br><span class="line"><span class="comment">* @return 0,成功; -1,失败,并且设置 errno 的值为相应的错误码。</span></span><br><span class="line"><span class="comment">*/</span></span><br><span class="line"> ><span class="function"><span class="keyword">int</span> <span class="title">epoll_ctl</span><span class="params">(<span class="keyword">int</span> epfd, <span class="keyword">int</span> op, <span class="keyword">int</span> fd, struct epoll_event *event)</span></span>;</span><br><span class="line"></span><br><span class="line"> ><span class="comment">/* op 可以选择的设置 */</span></span><br><span class="line"> >EPOLL_CTL_ADD <span class="comment">// 向 epfd 添加文件参数 fd 表示的描述符。</span></span><br><span class="line"> >EPOLL_CTL_MOD <span class="comment">// 修改参数 fd 的 event 事件。</span></span><br><span class="line"> >EPOLL_CTL_DEL <span class="comment">// 从 epfd 中删除 fd 描述符。</span></span><br><span class="line"></span><br><span class="line"> ><span class="comment">/* epoll_event 结构体 */</span></span><br><span class="line"> ><span class="class"><span class="keyword">struct</span> <span class="title">epoll_event</span> {</span></span><br><span class="line"> <span class="keyword">uint32_t</span> events; <span class="comment">/* epoll 事件 */</span></span><br><span class="line"> <span class="keyword">epoll_data_t</span> data; <span class="comment">/* 用户数据 */</span></span><br><span class="line"> >};</span><br></pre></td></tr></table></figure><p>结构体 <code>epoll_event</code> 的 <code>events</code> 成员变量表示要监视的事件,可选的事件如下,彼此之间可以进行或操作:</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line">>EPOLLIN <span class="comment">// 有数据可以读取。</span></span><br><span class="line">>EPOLLOUT <span class="comment">// 可以写数据。</span></span><br><span class="line">>EPOLLPRI <span class="comment">// 有紧急的数据需要读取。</span></span><br><span class="line">>EPOLLERR <span class="comment">// 指定的文件描述符发生错误。</span></span><br><span class="line">>EPOLLHUP <span class="comment">// 指定的文件描述符挂起。</span></span><br><span class="line">>EPOLLET <span class="comment">// 设置 epoll 为边沿触发,默认触发模式为水平触发。</span></span><br><span class="line">>EPOLLONESHOT <span class="comment">// 一次性的监视,当监视完成以后还需要再次监视某个 fd,那么就需要将 fd 重新添加到 epoll 里面。</span></span><br></pre></td></tr></table></figure><p>一切都设置好以后应用程序就可以通过 <code>epoll_wait</code> 函数来等待事件的发生,类似 <code>select</code> 函数。</p><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br></pre></td><td class="code"><pre><span class="line"> ><span class="comment">/**</span></span><br><span class="line"><span class="comment">* @param epfd 要等待的 epoll</span></span><br><span class="line"><span class="comment">* @param events 指向 epoll_event 结构体的数组,当有事件发生的时候 Linux 内核会填写 events,调</span></span><br><span class="line"><span class="comment"> >用者可以根据 events 判断发生了哪些事件。</span></span><br><span class="line"><span class="comment">* @param maxevents events 数组大小,必须大于0</span></span><br><span class="line"><span class="comment">* @param timeout 超时时间,单位为ms。</span></span><br><span class="line"><span class="comment">* @return 0,超时;-1,错误;其他值,准备就绪的文件描述符数量。</span></span><br><span class="line"><span class="comment">*/</span></span><br><span class="line"> ><span class="function"><span class="keyword">int</span> <span class="title">epoll_wait</span><span class="params">(<span class="keyword">int</span> epfd, struct epoll_event *events, <span class="keyword">int</span> maxevents, <span class="keyword">int</span> timeout)</span></span>;</span><br></pre></td></tr></table></figure><p><strong><code>epoll</code> 更多的是用在大规模的并发服务器上,因为在这种场合下 <code>select</code> 和 <code>poll</code> 并不适合,当设计到的文件描述符(<code>fd</code>)比较少的时候就适合用 <code>selcet</code> 和 <code>poll</code>。</strong></p></blockquote></li></ol><h2 id="二、编写试验驱动"><a href="#二、编写试验驱动" class="headerlink" title="二、编写试验驱动"></a>二、编写试验驱动</h2><h3 id="2-1-阻塞式访问驱动"><a href="#2-1-阻塞式访问驱动" class="headerlink" title="2.1 阻塞式访问驱动"></a>2.1 阻塞式访问驱动</h3><ol><li>等待事件</li></ol><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br><span class="line">39</span><br><span class="line">40</span><br><span class="line">41</span><br><span class="line">42</span><br><span class="line">43</span><br><span class="line">44</span><br><span class="line">45</span><br><span class="line">46</span><br><span class="line">47</span><br><span class="line">48</span><br><span class="line">49</span><br><span class="line">50</span><br><span class="line">51</span><br><span class="line">52</span><br><span class="line">53</span><br><span class="line">54</span><br><span class="line">55</span><br><span class="line">56</span><br><span class="line">57</span><br><span class="line">58</span><br><span class="line">59</span><br><span class="line">60</span><br><span class="line">61</span><br><span class="line">62</span><br><span class="line">63</span><br><span class="line">64</span><br><span class="line">65</span><br><span class="line">66</span><br><span class="line">67</span><br><span class="line">68</span><br><span class="line">69</span><br><span class="line">70</span><br><span class="line">71</span><br><span class="line">72</span><br><span class="line">73</span><br><span class="line">74</span><br><span class="line">75</span><br><span class="line">76</span><br><span class="line">77</span><br><span class="line">78</span><br><span class="line">79</span><br><span class="line">80</span><br><span class="line">81</span><br><span class="line">82</span><br><span class="line">83</span><br><span class="line">84</span><br><span class="line">85</span><br><span class="line">86</span><br><span class="line">87</span><br><span class="line">88</span><br><span class="line">89</span><br><span class="line">90</span><br><span class="line">91</span><br><span class="line">92</span><br><span class="line">93</span><br><span class="line">94</span><br><span class="line">95</span><br><span class="line">96</span><br><span class="line">97</span><br><span class="line">98</span><br><span class="line">99</span><br><span class="line">100</span><br><span class="line">101</span><br><span class="line">102</span><br><span class="line">103</span><br><span class="line">104</span><br><span class="line">105</span><br><span class="line">106</span><br><span class="line">107</span><br><span class="line">108</span><br><span class="line">109</span><br><span class="line">110</span><br><span class="line">111</span><br><span class="line">112</span><br><span class="line">113</span><br><span class="line">114</span><br><span class="line">115</span><br><span class="line">116</span><br><span class="line">117</span><br><span class="line">118</span><br><span class="line">119</span><br><span class="line">120</span><br><span class="line">121</span><br><span class="line">122</span><br><span class="line">123</span><br><span class="line">124</span><br><span class="line">125</span><br><span class="line">126</span><br><span class="line">127</span><br><span class="line">128</span><br><span class="line">129</span><br><span class="line">130</span><br><span class="line">131</span><br><span class="line">132</span><br><span class="line">133</span><br><span class="line">134</span><br><span class="line">135</span><br><span class="line">136</span><br><span class="line">137</span><br><span class="line">138</span><br><span class="line">139</span><br><span class="line">140</span><br><span class="line">141</span><br><span class="line">142</span><br><span class="line">143</span><br><span class="line">144</span><br><span class="line">145</span><br><span class="line">146</span><br><span class="line">147</span><br><span class="line">148</span><br><span class="line">149</span><br><span class="line">150</span><br><span class="line">151</span><br><span class="line">152</span><br><span class="line">153</span><br><span class="line">154</span><br><span class="line">155</span><br><span class="line">156</span><br><span class="line">157</span><br><span class="line">158</span><br><span class="line">159</span><br><span class="line">160</span><br><span class="line">161</span><br><span class="line">162</span><br><span class="line">163</span><br><span class="line">164</span><br><span class="line">165</span><br><span class="line">166</span><br><span class="line">167</span><br><span class="line">168</span><br><span class="line">169</span><br><span class="line">170</span><br><span class="line">171</span><br><span class="line">172</span><br><span class="line">173</span><br><span class="line">174</span><br><span class="line">175</span><br><span class="line">176</span><br><span class="line">177</span><br><span class="line">178</span><br><span class="line">179</span><br><span class="line">180</span><br><span class="line">181</span><br><span class="line">182</span><br><span class="line">183</span><br><span class="line">184</span><br><span class="line">185</span><br><span class="line">186</span><br><span class="line">187</span><br><span class="line">188</span><br><span class="line">189</span><br><span class="line">190</span><br><span class="line">191</span><br><span class="line">192</span><br><span class="line">193</span><br><span class="line">194</span><br><span class="line">195</span><br><span class="line">196</span><br><span class="line">197</span><br><span class="line">198</span><br><span class="line">199</span><br><span class="line">200</span><br><span class="line">201</span><br><span class="line">202</span><br><span class="line">203</span><br><span class="line">204</span><br><span class="line">205</span><br><span class="line">206</span><br><span class="line">207</span><br><span class="line">208</span><br><span class="line">209</span><br><span class="line">210</span><br><span class="line">211</span><br><span class="line">212</span><br><span class="line">213</span><br><span class="line">214</span><br><span class="line">215</span><br><span class="line">216</span><br><span class="line">217</span><br><span class="line">218</span><br><span class="line">219</span><br><span class="line">220</span><br><span class="line">221</span><br><span class="line">222</span><br><span class="line">223</span><br><span class="line">224</span><br><span class="line">225</span><br><span class="line">226</span><br><span class="line">227</span><br><span class="line">228</span><br><span class="line">229</span><br><span class="line">230</span><br><span class="line">231</span><br><span class="line">232</span><br><span class="line">233</span><br><span class="line">234</span><br><span class="line">235</span><br><span class="line">236</span><br><span class="line">237</span><br><span class="line">238</span><br><span class="line">239</span><br><span class="line">240</span><br><span class="line">241</span><br><span class="line">242</span><br><span class="line">243</span><br><span class="line">244</span><br><span class="line">245</span><br><span class="line">246</span><br><span class="line">247</span><br><span class="line">248</span><br><span class="line">249</span><br><span class="line">250</span><br><span class="line">251</span><br><span class="line">252</span><br><span class="line">253</span><br><span class="line">254</span><br><span class="line">255</span><br><span class="line">256</span><br><span class="line">257</span><br><span class="line">258</span><br><span class="line">259</span><br><span class="line">260</span><br><span class="line">261</span><br><span class="line">262</span><br><span class="line">263</span><br><span class="line">264</span><br><span class="line">265</span><br><span class="line">266</span><br><span class="line">267</span><br><span class="line">268</span><br><span class="line">269</span><br><span class="line">270</span><br><span class="line">271</span><br><span class="line">272</span><br><span class="line">273</span><br><span class="line">274</span><br><span class="line">275</span><br><span class="line">276</span><br><span class="line">277</span><br><span class="line">278</span><br><span class="line">279</span><br><span class="line">280</span><br><span class="line">281</span><br><span class="line">282</span><br><span class="line">283</span><br><span class="line">284</span><br><span class="line">285</span><br><span class="line">286</span><br><span class="line">287</span><br><span class="line">288</span><br><span class="line">289</span><br><span class="line">290</span><br><span class="line">291</span><br><span class="line">292</span><br><span class="line">293</span><br><span class="line">294</span><br><span class="line">295</span><br><span class="line">296</span><br><span class="line">297</span><br><span class="line">298</span><br><span class="line">299</span><br><span class="line">300</span><br><span class="line">301</span><br><span class="line">302</span><br><span class="line">303</span><br><span class="line">304</span><br><span class="line">305</span><br><span class="line">306</span><br><span class="line">307</span><br><span class="line">308</span><br><span class="line">309</span><br><span class="line">310</span><br><span class="line">311</span><br><span class="line">312</span><br><span class="line">313</span><br><span class="line">314</span><br><span class="line">315</span><br><span class="line">316</span><br><span class="line">317</span><br><span class="line">318</span><br><span class="line">319</span><br><span class="line">320</span><br><span class="line">321</span><br><span class="line">322</span><br><span class="line">323</span><br><span class="line">324</span><br><span class="line">325</span><br><span class="line">326</span><br><span class="line">327</span><br><span class="line">328</span><br><span class="line">329</span><br><span class="line">330</span><br><span class="line">331</span><br><span class="line">332</span><br><span class="line">333</span><br><span class="line">334</span><br><span class="line">335</span><br><span class="line">336</span><br><span class="line">337</span><br><span class="line">338</span><br><span class="line">339</span><br><span class="line">340</span><br><span class="line">341</span><br><span class="line">342</span><br><span class="line">343</span><br><span class="line">344</span><br><span class="line">345</span><br><span class="line">346</span><br><span class="line">347</span><br><span class="line">348</span><br><span class="line">349</span><br><span class="line">350</span><br><span class="line">351</span><br><span class="line">352</span><br><span class="line">353</span><br><span class="line">354</span><br><span class="line">355</span><br><span class="line">356</span><br><span class="line">357</span><br><span class="line">358</span><br><span class="line">359</span><br><span class="line">360</span><br><span class="line">361</span><br><span class="line">362</span><br><span class="line">363</span><br><span class="line">364</span><br><span class="line">365</span><br><span class="line">366</span><br><span class="line">367</span><br><span class="line">368</span><br><span class="line">369</span><br><span class="line">370</span><br><span class="line">371</span><br><span class="line">372</span><br><span class="line">373</span><br><span class="line">374</span><br><span class="line">375</span><br><span class="line">376</span><br><span class="line">377</span><br><span class="line">378</span><br><span class="line">379</span><br><span class="line">380</span><br><span class="line">381</span><br><span class="line">382</span><br><span class="line">383</span><br><span class="line">384</span><br><span class="line">385</span><br></pre></td><td class="code"><pre><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/types.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/kernel.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/delay.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/init.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/ide.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/module.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/errno.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/gpio.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/cdev.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of_address.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of_irq.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of_gpio.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/timer.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/jiffies.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/string.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/irq.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/interrupt.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><asm/mach/map.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><asm/uaccess.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><asm/io.h></span></span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEYIRQ_CNT 1</span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEYIRQ_NAME <span class="meta-string">"keyirq"</span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEY_NUM 1</span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEY0_VALUE 0X01</span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> INVAKEY 0XFF <span class="comment">/* 无效按键值 */</span></span></span><br><span class="line"></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">key_dev</span></span></span><br><span class="line"><span class="class">{</span></span><br><span class="line"> <span class="keyword">int</span> gpio; <span class="comment">/* IO编号 */</span></span><br><span class="line"> <span class="keyword">int</span> irqNum; <span class="comment">/* 中断号 */</span></span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">char</span> value; <span class="comment">/* 按键值 */</span></span><br><span class="line"> <span class="keyword">char</span> name[<span class="number">10</span>]; <span class="comment">/* 名称 */</span></span><br><span class="line"> <span class="keyword">irqreturn_t</span> (*handler)(<span class="keyword">int</span>, <span class="keyword">void</span>*); <span class="comment">/* 中断处理函数 */</span></span><br><span class="line"></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">tasklet_struct</span> <span class="title">tasklet</span>;</span></span><br><span class="line">};</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment">/* 定义设备驱动结构体 */</span></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span></span></span><br><span class="line"><span class="class">{</span></span><br><span class="line"> <span class="keyword">dev_t</span> devid; <span class="comment">/* 设备ID */</span></span><br><span class="line"> <span class="keyword">int</span> major; <span class="comment">/* 主设备号 */</span></span><br><span class="line"> <span class="keyword">int</span> minor; <span class="comment">/* 次设备号 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">cdev</span> <span class="title">cdev</span>;</span> <span class="comment">/* 字符设备结构体 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">class</span> *<span class="title">class</span>;</span> <span class="comment">/* 设备类 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">device</span> *<span class="title">device</span>;</span> <span class="comment">/* 设备节点 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">device_node</span> *<span class="title">nd</span>;</span> <span class="comment">/* 设备树节点 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">key_dev</span> <span class="title">key</span>[<span class="title">KEY_NUM</span>];</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">timer_list</span> <span class="title">timer</span>;</span> <span class="comment">/* 定时器 */</span></span><br><span class="line"></span><br><span class="line"> <span class="keyword">atomic_t</span> keyValue; <span class="comment">/* 按键值 */</span></span><br><span class="line"> <span class="keyword">atomic_t</span> keyRelease; <span class="comment">/* 按键是否释放 */</span></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">wait_queue_head_t</span> r_wait; <span class="comment">/* 读等待队列头 */</span></span><br><span class="line">};</span><br><span class="line"></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> <span class="title">keyirq</span>;</span> <span class="comment">/* 定义一个keyirq设备 */</span></span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> <span class="title">keyirq_open</span><span class="params">(struct inode *inode, struct file *filp)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> filp->private_data = &keyirq;</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line">} </span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> <span class="title">keyirq_release</span><span class="params">(struct inode *inode, struct file *filp)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">filp</span>-><span class="title">private_data</span>;</span></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">ssize_t</span> <span class="title">keyirq_write</span><span class="params">(struct file *filp, <span class="keyword">const</span> <span class="keyword">char</span> __user *buf, <span class="keyword">size_t</span> count, <span class="keyword">loff_t</span> *ppos)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">filp</span>-><span class="title">private_data</span>;</span></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">ssize_t</span> <span class="title">keyirq_read</span><span class="params">(struct file *filp, <span class="keyword">char</span> __user *buf, <span class="keyword">size_t</span> count, <span class="keyword">loff_t</span> *ppos)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">filp</span>-><span class="title">private_data</span>;</span></span><br><span class="line"> <span class="keyword">int</span> ret = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">char</span> keyValue;</span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">char</span> keyRelease;</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 等待事件 */</span></span><br><span class="line"> wait_event_interruptible(dev->r_wait, atomic_read(&dev->keyRelease));</span><br><span class="line"> </span><br><span class="line"> <span class="comment">// <span class="doctag">NOTE:</span>wait_event 不可以被信号打断,使用 kill -9 PID 无法杀掉任务</span></span><br><span class="line"> <span class="comment">// wait_event(dev->r_wait, atomic_read(&dev->keyRelease)); /* 等待按键值有效 */</span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"> keyValue = atomic_read(&dev->keyValue);</span><br><span class="line"> keyRelease = atomic_read(&dev->keyRelease);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (keyRelease)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 一次有效按键 */</span></span><br><span class="line"> <span class="keyword">if</span> (keyValue & <span class="number">0X80</span>)</span><br><span class="line"> {</span><br><span class="line"> keyValue &= ~<span class="number">0X80</span>; <span class="comment">// 去除标记</span></span><br><span class="line"> ret = copy_to_user(buf, &keyValue, <span class="keyword">sizeof</span>(keyValue));</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> data_err;</span><br><span class="line"> }</span><br><span class="line"> atomic_set(&dev->keyRelease, <span class="number">0</span>);</span><br><span class="line"> } </span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> data_err;</span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"></span><br><span class="line"> <span class="keyword">return</span> ret;</span><br><span class="line">data_err:</span><br><span class="line"> <span class="keyword">return</span> -EINVAL;</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/* 操作集 */</span></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">file_operations</span> <span class="title">keyirq_fops</span> = {</span></span><br><span class="line"> .owner = THIS_MODULE,</span><br><span class="line"> .<span class="built_in">open</span> = keyirq_open,</span><br><span class="line"> .<span class="built_in">release</span> = keyirq_release,</span><br><span class="line"> .<span class="built_in">write</span> = keyirq_write,</span><br><span class="line"> .<span class="built_in">read</span> = keyirq_read,</span><br><span class="line">};</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 定时器处理函数</span></span><br><span class="line"><span class="comment"> * @param arg 用户参数</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">void</span> <span class="title">timer_func</span><span class="params">(<span class="keyword">unsigned</span> <span class="keyword">long</span> arg)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = (<span class="title">struct</span> <span class="title">keyirq_dev</span>*)<span class="title">arg</span>;</span></span><br><span class="line"> <span class="keyword">int</span> value = <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line"> value = gpio_get_value(dev->key[<span class="number">0</span>].gpio);</span><br><span class="line"> <span class="keyword">if</span> (value == <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">// 按键按下</span></span><br><span class="line"> printk(<span class="string">"KEY0 Press!\r\n"</span>);</span><br><span class="line"> atomic_set(&dev->keyValue, dev->key[<span class="number">0</span>].value);</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span> <span class="keyword">if</span>(value == <span class="number">1</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">// 按键释放</span></span><br><span class="line"> printk(<span class="string">"KEY0 Release!\r\n"</span>);</span><br><span class="line"> atomic_set(&dev->keyValue, dev->key[<span class="number">0</span>].value | <span class="number">0X80</span>); <span class="comment">// 打上标签,标记按键按下</span></span><br><span class="line"> atomic_set(&dev->keyRelease, <span class="number">1</span>);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 唤醒进程 */</span></span><br><span class="line"> <span class="keyword">if</span>(atomic_read(&dev->keyRelease))</span><br><span class="line"> {</span><br><span class="line"> wake_up(&dev->r_wait);</span><br><span class="line"> }</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief key0 tasklet 处理函数</span></span><br><span class="line"><span class="comment"> * @param arg 传递的参数</span></span><br><span class="line"><span class="comment"> */</span> </span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">void</span> <span class="title">key0_tasklet</span><span class="params">(<span class="keyword">unsigned</span> <span class="keyword">long</span> arg)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = (<span class="title">struct</span> <span class="title">keyirq_dev</span> *)<span class="title">arg</span>;</span></span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 启动定时器来延时消抖 */</span></span><br><span class="line"> dev->timer.data = (<span class="keyword">volatile</span> <span class="keyword">unsigned</span> <span class="keyword">long</span>)arg;</span><br><span class="line"> mod_timer(&dev->timer, jiffies + msecs_to_jiffies(<span class="number">20</span>));</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 按键中断处理函数</span></span><br><span class="line"><span class="comment"> * @param irq 中断号</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">irqreturn_t</span> <span class="title">key0_handler</span><span class="params">(<span class="keyword">int</span> irq, <span class="keyword">void</span> *arg)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">arg</span>;</span></span><br><span class="line"></span><br><span class="line"> tasklet_schedule(&dev->key[<span class="number">0</span>].tasklet); <span class="comment">/* 调度对应的tasklet */</span></span><br><span class="line"> <span class="keyword">return</span> IRQ_HANDLED;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 按键初始化</span></span><br><span class="line"><span class="comment"> * @param dev 设备结构体</span></span><br><span class="line"><span class="comment"> * @return 错误类型</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> <span class="title">keyio_init</span><span class="params">(struct keyirq_dev *dev)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> ret = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">int</span> i = <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 按键初始化 */</span></span><br><span class="line"> dev->nd = of_find_node_by_path(<span class="string">"/key"</span>);</span><br><span class="line"> <span class="keyword">if</span> (dev->nd == <span class="literal">NULL</span>)</span><br><span class="line"> {</span><br><span class="line"> ret = -EINVAL;</span><br><span class="line"> <span class="keyword">goto</span> fail_nd;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 可能有多个按键,因此需要根据实际数量来读取 */</span></span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> dev->key[i].gpio = of_get_named_gpio(dev->nd, <span class="string">"key-gpios"</span>, i);</span><br><span class="line"> <span class="keyword">if</span> (dev->key[i].gpio < <span class="number">0</span> )</span><br><span class="line"> {</span><br><span class="line"> ret = -EINVAL;</span><br><span class="line"> <span class="keyword">goto</span> fail_getGpio;</span><br><span class="line"> }</span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> <span class="built_in">memset</span>(dev->key[i].name, <span class="number">0</span>, <span class="keyword">sizeof</span>(dev->key[i].name));</span><br><span class="line"> <span class="built_in">sprintf</span>(dev->key[i].name, <span class="string">"KEY%d"</span>, i);</span><br><span class="line"> ret = gpio_request(dev->key[i].gpio, dev->key[i].name);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span> )</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_requestGpio;</span><br><span class="line"> }</span><br><span class="line"> ret = gpio_direction_input(dev->key[i].gpio);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_setGpioDir;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 获取中断号,两种方式皆可 */</span></span><br><span class="line"> dev->key[i].irqNum = gpio_to_irq(dev->key[i].gpio);</span><br><span class="line"> <span class="comment">// dev->key[i].irqNum = irq_of_parse_and_map(dev->nd, i);</span></span><br><span class="line"></span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 按键中断初始化 */</span></span><br><span class="line"> dev->key[<span class="number">0</span>].handler = key0_handler;</span><br><span class="line"> dev->key[<span class="number">0</span>].value = KEY0_VALUE;</span><br><span class="line"></span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 跳变沿触发方式 */</span></span><br><span class="line"> ret = request_irq(dev->key[i].irqNum, dev->key[i].handler, IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING, dev->key[i].name, dev);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> printk(<span class="string">"irq %d request failed!\r\n"</span>, dev->key[i].irqNum);</span><br><span class="line"> <span class="keyword">goto</span> fail_irq;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* <span class="doctag">NOTE:</span>没有根据多个按键来初始化各自的tasklet,实际中需要根据情况来编写 */</span></span><br><span class="line"> tasklet_init(&dev->key[i].tasklet, key0_tasklet, (<span class="keyword">unsigned</span> <span class="keyword">long</span>)dev);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化定时器 */</span></span><br><span class="line"> init_timer(&dev->timer);</span><br><span class="line"> dev->timer.function = timer_func;</span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line">fail_irq:</span><br><span class="line">fail_setGpioDir:</span><br><span class="line"> <span class="keyword">for</span> (i = i<span class="number">-1</span>; i >= <span class="number">0</span>; i--)</span><br><span class="line"> {</span><br><span class="line"> gpio_free(dev->key[i].gpio);</span><br><span class="line"> }</span><br><span class="line">fail_requestGpio:</span><br><span class="line">fail_getGpio:</span><br><span class="line">fail_nd:</span><br><span class="line"> <span class="keyword">return</span> ret;</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment">/* 入口和出口 */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> __init <span class="title">keyirq_init</span><span class="params">(<span class="keyword">void</span>)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> ret = <span class="number">0</span>;</span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 注册字符设备ID */</span></span><br><span class="line"> keyirq.major = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">if</span>(keyirq.major)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 给定设备号 */</span></span><br><span class="line"> keyirq.devid = MKDEV(keyirq.major, <span class="number">0</span>);</span><br><span class="line"> ret = register_chrdev_region(keyirq.devid, KEYIRQ_CNT, KEYIRQ_NAME);</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> alloc_chrdev_region(&keyirq.devid, <span class="number">0</span>,KEYIRQ_CNT,KEYIRQ_NAME);</span><br><span class="line"> keyirq.major = MAJOR(keyirq.devid);</span><br><span class="line"> keyirq.minor = MINOR(keyirq.devid);</span><br><span class="line"> printk(<span class="string">"dev Major ID:%d\r\n"</span>,keyirq.major);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_devid;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化字符设备 */</span></span><br><span class="line"> keyirq.cdev.owner = THIS_MODULE;</span><br><span class="line"> cdev_init(&keyirq.cdev, &keyirq_fops);</span><br><span class="line"> ret = cdev_add(&keyirq.cdev, keyirq.devid, KEYIRQ_CNT);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_cdev;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 创建设备类 */</span></span><br><span class="line"> keyirq<span class="class">.<span class="keyword">class</span> = <span class="title">class_create</span>(<span class="title">THIS_MODULE</span>, <span class="title">KEYIRQ_NAME</span>);</span></span><br><span class="line"> <span class="keyword">if</span> (IS_ERR(keyirq.class))</span><br><span class="line"> {</span><br><span class="line"> ret = PTR_ERR(keyirq.class);</span><br><span class="line"> <span class="keyword">goto</span> fail_class;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 创建设备节点 */</span></span><br><span class="line"> keyirq.device = device_create(keyirq.class, <span class="literal">NULL</span>, keyirq.devid, <span class="literal">NULL</span>, KEYIRQ_NAME);</span><br><span class="line"> <span class="keyword">if</span> (IS_ERR(keyirq.device))</span><br><span class="line"> {</span><br><span class="line"> ret = PTR_ERR(keyirq.device);</span><br><span class="line"> <span class="keyword">goto</span> fail_device;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化IO */</span></span><br><span class="line"> ret = keyio_init(&keyirq);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_init;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化原子变量 */</span></span><br><span class="line"> atomic_set(&keyirq.keyValue, INVAKEY);</span><br><span class="line"> atomic_set(&keyirq.keyRelease, <span class="number">0</span>);</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化等待队列头 */</span></span><br><span class="line"> init_waitqueue_head(&keyirq.r_wait);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line">fail_init:</span><br><span class="line"> device_destroy(keyirq.class, keyirq.devid);</span><br><span class="line">fail_device:</span><br><span class="line"> class_destroy(keyirq.class);</span><br><span class="line">fail_class:</span><br><span class="line"> cdev_del(&keyirq.cdev);</span><br><span class="line">fail_cdev:</span><br><span class="line"> unregister_chrdev_region(keyirq.devid, KEYIRQ_CNT);</span><br><span class="line">fail_devid:</span><br><span class="line"> <span class="keyword">return</span> ret;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">void</span> __exit <span class="title">keyirq_exit</span><span class="params">(<span class="keyword">void</span>)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> i = <span class="number">0</span>;</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 释放中断和IO */</span></span><br><span class="line"> free_irq(keyirq.key[i].irqNum, &keyirq);</span><br><span class="line"> gpio_free(keyirq.key[i].gpio);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 删除定时器 */</span></span><br><span class="line"> del_timer_sync(&keyirq.timer);</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 注销字符设备驱动 */</span></span><br><span class="line"> device_destroy(keyirq.class, keyirq.devid);</span><br><span class="line"> class_destroy(keyirq.class);</span><br><span class="line"> cdev_del(&keyirq.cdev);</span><br><span class="line"> unregister_chrdev_region(keyirq.devid,KEYIRQ_CNT);</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/* 注册驱动和卸载驱动 */</span></span><br><span class="line">module_init(keyirq_init);</span><br><span class="line">module_exit(keyirq_exit);</span><br><span class="line"></span><br><span class="line">MODULE_LICENSE(<span class="string">"GPL"</span>);</span><br><span class="line">MODULE_AUTHOR(<span class="string">"fengyuhang"</span>);</span><br></pre></td></tr></table></figure><ol start="2"><li>等待队列项</li></ol><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br><span class="line">39</span><br><span class="line">40</span><br><span class="line">41</span><br><span class="line">42</span><br><span class="line">43</span><br><span class="line">44</span><br><span class="line">45</span><br><span class="line">46</span><br><span class="line">47</span><br><span class="line">48</span><br><span class="line">49</span><br><span class="line">50</span><br><span class="line">51</span><br><span class="line">52</span><br><span class="line">53</span><br><span class="line">54</span><br><span class="line">55</span><br><span class="line">56</span><br><span class="line">57</span><br><span class="line">58</span><br><span class="line">59</span><br><span class="line">60</span><br><span class="line">61</span><br><span class="line">62</span><br><span class="line">63</span><br><span class="line">64</span><br><span class="line">65</span><br><span class="line">66</span><br><span class="line">67</span><br><span class="line">68</span><br><span class="line">69</span><br><span class="line">70</span><br><span class="line">71</span><br><span class="line">72</span><br><span class="line">73</span><br><span class="line">74</span><br><span class="line">75</span><br><span class="line">76</span><br><span class="line">77</span><br><span class="line">78</span><br><span class="line">79</span><br><span class="line">80</span><br><span class="line">81</span><br><span class="line">82</span><br><span class="line">83</span><br><span class="line">84</span><br><span class="line">85</span><br><span class="line">86</span><br><span class="line">87</span><br><span class="line">88</span><br><span class="line">89</span><br><span class="line">90</span><br><span class="line">91</span><br><span class="line">92</span><br><span class="line">93</span><br><span class="line">94</span><br><span class="line">95</span><br><span class="line">96</span><br><span class="line">97</span><br><span class="line">98</span><br><span class="line">99</span><br><span class="line">100</span><br><span class="line">101</span><br><span class="line">102</span><br><span class="line">103</span><br><span class="line">104</span><br><span class="line">105</span><br><span class="line">106</span><br><span class="line">107</span><br><span class="line">108</span><br><span class="line">109</span><br><span class="line">110</span><br><span class="line">111</span><br><span class="line">112</span><br><span class="line">113</span><br><span class="line">114</span><br><span class="line">115</span><br><span class="line">116</span><br><span class="line">117</span><br><span class="line">118</span><br><span class="line">119</span><br><span class="line">120</span><br><span class="line">121</span><br><span class="line">122</span><br><span class="line">123</span><br><span class="line">124</span><br><span class="line">125</span><br><span class="line">126</span><br><span class="line">127</span><br><span class="line">128</span><br><span class="line">129</span><br><span class="line">130</span><br><span class="line">131</span><br><span class="line">132</span><br><span class="line">133</span><br><span class="line">134</span><br><span class="line">135</span><br><span class="line">136</span><br><span class="line">137</span><br><span class="line">138</span><br><span class="line">139</span><br><span class="line">140</span><br><span class="line">141</span><br><span class="line">142</span><br><span class="line">143</span><br><span class="line">144</span><br><span class="line">145</span><br><span class="line">146</span><br><span class="line">147</span><br><span class="line">148</span><br><span class="line">149</span><br><span class="line">150</span><br><span class="line">151</span><br><span class="line">152</span><br><span class="line">153</span><br><span class="line">154</span><br><span class="line">155</span><br><span class="line">156</span><br><span class="line">157</span><br><span class="line">158</span><br><span class="line">159</span><br><span class="line">160</span><br><span class="line">161</span><br><span class="line">162</span><br><span class="line">163</span><br><span class="line">164</span><br><span class="line">165</span><br><span class="line">166</span><br><span class="line">167</span><br><span class="line">168</span><br><span class="line">169</span><br><span class="line">170</span><br><span class="line">171</span><br><span class="line">172</span><br><span class="line">173</span><br><span class="line">174</span><br><span class="line">175</span><br><span class="line">176</span><br><span class="line">177</span><br><span class="line">178</span><br><span class="line">179</span><br><span class="line">180</span><br><span class="line">181</span><br><span class="line">182</span><br><span class="line">183</span><br><span class="line">184</span><br><span class="line">185</span><br><span class="line">186</span><br><span class="line">187</span><br><span class="line">188</span><br><span class="line">189</span><br><span class="line">190</span><br><span class="line">191</span><br><span class="line">192</span><br><span class="line">193</span><br><span class="line">194</span><br><span class="line">195</span><br><span class="line">196</span><br><span class="line">197</span><br><span class="line">198</span><br><span class="line">199</span><br><span class="line">200</span><br><span class="line">201</span><br><span class="line">202</span><br><span class="line">203</span><br><span class="line">204</span><br><span class="line">205</span><br><span class="line">206</span><br><span class="line">207</span><br><span class="line">208</span><br><span class="line">209</span><br><span class="line">210</span><br><span class="line">211</span><br><span class="line">212</span><br><span class="line">213</span><br><span class="line">214</span><br><span class="line">215</span><br><span class="line">216</span><br><span class="line">217</span><br><span class="line">218</span><br><span class="line">219</span><br><span class="line">220</span><br><span class="line">221</span><br><span class="line">222</span><br><span class="line">223</span><br><span class="line">224</span><br><span class="line">225</span><br><span class="line">226</span><br><span class="line">227</span><br><span class="line">228</span><br><span class="line">229</span><br><span class="line">230</span><br><span class="line">231</span><br><span class="line">232</span><br><span class="line">233</span><br><span class="line">234</span><br><span class="line">235</span><br><span class="line">236</span><br><span class="line">237</span><br><span class="line">238</span><br><span class="line">239</span><br><span class="line">240</span><br><span class="line">241</span><br><span class="line">242</span><br><span class="line">243</span><br><span class="line">244</span><br><span class="line">245</span><br><span class="line">246</span><br><span class="line">247</span><br><span class="line">248</span><br><span class="line">249</span><br><span class="line">250</span><br><span class="line">251</span><br><span class="line">252</span><br><span class="line">253</span><br><span class="line">254</span><br><span class="line">255</span><br><span class="line">256</span><br><span class="line">257</span><br><span class="line">258</span><br><span class="line">259</span><br><span class="line">260</span><br><span class="line">261</span><br><span class="line">262</span><br><span class="line">263</span><br><span class="line">264</span><br><span class="line">265</span><br><span class="line">266</span><br><span class="line">267</span><br><span class="line">268</span><br><span class="line">269</span><br><span class="line">270</span><br><span class="line">271</span><br><span class="line">272</span><br><span class="line">273</span><br><span class="line">274</span><br><span class="line">275</span><br><span class="line">276</span><br><span class="line">277</span><br><span class="line">278</span><br><span class="line">279</span><br><span class="line">280</span><br><span class="line">281</span><br><span class="line">282</span><br><span class="line">283</span><br><span class="line">284</span><br><span class="line">285</span><br><span class="line">286</span><br><span class="line">287</span><br><span class="line">288</span><br><span class="line">289</span><br><span class="line">290</span><br><span class="line">291</span><br><span class="line">292</span><br><span class="line">293</span><br><span class="line">294</span><br><span class="line">295</span><br><span class="line">296</span><br><span class="line">297</span><br><span class="line">298</span><br><span class="line">299</span><br><span class="line">300</span><br><span class="line">301</span><br><span class="line">302</span><br><span class="line">303</span><br><span class="line">304</span><br><span class="line">305</span><br><span class="line">306</span><br><span class="line">307</span><br><span class="line">308</span><br><span class="line">309</span><br><span class="line">310</span><br><span class="line">311</span><br><span class="line">312</span><br><span class="line">313</span><br><span class="line">314</span><br><span class="line">315</span><br><span class="line">316</span><br><span class="line">317</span><br><span class="line">318</span><br><span class="line">319</span><br><span class="line">320</span><br><span class="line">321</span><br><span class="line">322</span><br><span class="line">323</span><br><span class="line">324</span><br><span class="line">325</span><br><span class="line">326</span><br><span class="line">327</span><br><span class="line">328</span><br><span class="line">329</span><br><span class="line">330</span><br><span class="line">331</span><br><span class="line">332</span><br><span class="line">333</span><br><span class="line">334</span><br><span class="line">335</span><br><span class="line">336</span><br><span class="line">337</span><br><span class="line">338</span><br><span class="line">339</span><br><span class="line">340</span><br><span class="line">341</span><br><span class="line">342</span><br><span class="line">343</span><br><span class="line">344</span><br><span class="line">345</span><br><span class="line">346</span><br><span class="line">347</span><br><span class="line">348</span><br><span class="line">349</span><br><span class="line">350</span><br><span class="line">351</span><br><span class="line">352</span><br><span class="line">353</span><br><span class="line">354</span><br><span class="line">355</span><br><span class="line">356</span><br><span class="line">357</span><br><span class="line">358</span><br><span class="line">359</span><br><span class="line">360</span><br><span class="line">361</span><br><span class="line">362</span><br><span class="line">363</span><br><span class="line">364</span><br><span class="line">365</span><br><span class="line">366</span><br><span class="line">367</span><br><span class="line">368</span><br><span class="line">369</span><br><span class="line">370</span><br><span class="line">371</span><br><span class="line">372</span><br><span class="line">373</span><br><span class="line">374</span><br><span class="line">375</span><br><span class="line">376</span><br><span class="line">377</span><br><span class="line">378</span><br><span class="line">379</span><br><span class="line">380</span><br><span class="line">381</span><br><span class="line">382</span><br><span class="line">383</span><br><span class="line">384</span><br><span class="line">385</span><br><span class="line">386</span><br><span class="line">387</span><br><span class="line">388</span><br><span class="line">389</span><br><span class="line">390</span><br><span class="line">391</span><br><span class="line">392</span><br><span class="line">393</span><br><span class="line">394</span><br><span class="line">395</span><br><span class="line">396</span><br><span class="line">397</span><br><span class="line">398</span><br></pre></td><td class="code"><pre><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/types.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/kernel.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/delay.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/init.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/ide.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/module.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/errno.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/gpio.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/cdev.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of_address.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of_irq.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of_gpio.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/timer.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/jiffies.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/string.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/irq.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/interrupt.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><asm/mach/map.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><asm/uaccess.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><asm/io.h></span></span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEYIRQ_CNT 1</span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEYIRQ_NAME <span class="meta-string">"keyirq"</span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEY_NUM 1</span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEY0_VALUE 0X01</span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> INVAKEY 0XFF <span class="comment">/* 无效按键值 */</span></span></span><br><span class="line"></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">key_dev</span></span></span><br><span class="line"><span class="class">{</span></span><br><span class="line"> <span class="keyword">int</span> gpio; <span class="comment">/* IO编号 */</span></span><br><span class="line"> <span class="keyword">int</span> irqNum; <span class="comment">/* 中断号 */</span></span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">char</span> value; <span class="comment">/* 按键值 */</span></span><br><span class="line"> <span class="keyword">char</span> name[<span class="number">10</span>]; <span class="comment">/* 名称 */</span></span><br><span class="line"> <span class="keyword">irqreturn_t</span> (*handler)(<span class="keyword">int</span>, <span class="keyword">void</span>*); <span class="comment">/* 中断处理函数 */</span></span><br><span class="line"></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">tasklet_struct</span> <span class="title">tasklet</span>;</span></span><br><span class="line">};</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment">/* 定义设备驱动结构体 */</span></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span></span></span><br><span class="line"><span class="class">{</span></span><br><span class="line"> <span class="keyword">dev_t</span> devid; <span class="comment">/* 设备ID */</span></span><br><span class="line"> <span class="keyword">int</span> major; <span class="comment">/* 主设备号 */</span></span><br><span class="line"> <span class="keyword">int</span> minor; <span class="comment">/* 次设备号 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">cdev</span> <span class="title">cdev</span>;</span> <span class="comment">/* 字符设备结构体 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">class</span> *<span class="title">class</span>;</span> <span class="comment">/* 设备类 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">device</span> *<span class="title">device</span>;</span> <span class="comment">/* 设备节点 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">device_node</span> *<span class="title">nd</span>;</span> <span class="comment">/* 设备树节点 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">key_dev</span> <span class="title">key</span>[<span class="title">KEY_NUM</span>];</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">timer_list</span> <span class="title">timer</span>;</span> <span class="comment">/* 定时器 */</span></span><br><span class="line"></span><br><span class="line"> <span class="keyword">atomic_t</span> keyValue; <span class="comment">/* 按键值 */</span></span><br><span class="line"> <span class="keyword">atomic_t</span> keyRelease; <span class="comment">/* 按键是否释放 */</span></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">wait_queue_head_t</span> r_wait; <span class="comment">/* 读等待队列头 */</span></span><br><span class="line">};</span><br><span class="line"></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> <span class="title">keyirq</span>;</span> <span class="comment">/* 定义一个keyirq设备 */</span></span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> <span class="title">keyirq_open</span><span class="params">(struct inode *inode, struct file *filp)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> filp->private_data = &keyirq;</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line">} </span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> <span class="title">keyirq_release</span><span class="params">(struct inode *inode, struct file *filp)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">filp</span>-><span class="title">private_data</span>;</span></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">ssize_t</span> <span class="title">keyirq_write</span><span class="params">(struct file *filp, <span class="keyword">const</span> <span class="keyword">char</span> __user *buf, <span class="keyword">size_t</span> count, <span class="keyword">loff_t</span> *ppos)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">filp</span>-><span class="title">private_data</span>;</span></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">ssize_t</span> <span class="title">keyirq_read</span><span class="params">(struct file *filp, <span class="keyword">char</span> __user *buf, <span class="keyword">size_t</span> count, <span class="keyword">loff_t</span> *ppos)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">filp</span>-><span class="title">private_data</span>;</span></span><br><span class="line"> <span class="keyword">int</span> ret = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">char</span> keyValue;</span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">char</span> keyRelease;</span><br><span class="line"></span><br><span class="line"><span class="meta">#<span class="meta-keyword">if</span> 0</span></span><br><span class="line"> <span class="comment">/* 等待事件 */</span></span><br><span class="line"> wait_event_interruptible(dev->r_wait, atomic_read(&dev->keyRelease)); <span class="comment">/* 等待按键值有效 */</span></span><br><span class="line"></span><br><span class="line"> <span class="comment">// <span class="doctag">NOTE:</span>wait_event 不可以被信号打断,使用 kill -9 PID 无法杀掉任务</span></span><br><span class="line"> <span class="comment">// wait_event(dev->r_wait, atomic_read(&dev->keyRelease)); /* 等待按键值有效 */</span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">endif</span></span></span><br><span class="line"> <span class="comment">/* 等待队列项 */</span></span><br><span class="line"> DECLARE_WAITQUEUE(wait, current); <span class="comment">/* 定义一个等待队列项 */</span></span><br><span class="line"></span><br><span class="line"> add_wait_queue(&dev->r_wait, &wait); <span class="comment">/* 将等待队列项添加到等待队列头中 */</span></span><br><span class="line"> __set_current_state(TASK_INTERRUPTIBLE); <span class="comment">/* 设置当前进程为可以被打断的状态 */</span></span><br><span class="line"> schedule(); <span class="comment">/* 切换 */</span></span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 唤醒后,进程从这里运行 */</span></span><br><span class="line"> <span class="keyword">if</span> (signal_pending(current)) <span class="comment">/* 判断当前进程是否有信号需要处理,返回值不为零表示有信号需要进行处理 */</span></span><br><span class="line"> {</span><br><span class="line"> ret = -ERESTARTSYS;</span><br><span class="line"> <span class="keyword">goto</span> data_err;</span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> keyValue = atomic_read(&dev->keyValue);</span><br><span class="line"> keyRelease = atomic_read(&dev->keyRelease);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (keyRelease)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 一次有效按键 */</span></span><br><span class="line"> <span class="keyword">if</span> (keyValue & <span class="number">0X80</span>)</span><br><span class="line"> {</span><br><span class="line"> keyValue &= ~<span class="number">0X80</span>; <span class="comment">// 去除标记</span></span><br><span class="line"> ret = copy_to_user(buf, &keyValue, <span class="keyword">sizeof</span>(keyValue));</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> data_err;</span><br><span class="line"> }</span><br><span class="line"> atomic_set(&dev->keyRelease, <span class="number">0</span>);</span><br><span class="line"> } </span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> data_err;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line">data_err:</span><br><span class="line"> __set_current_state(TASK_RUNNING); <span class="comment">/* 设置当前任务为运行状态 */</span></span><br><span class="line"> remove_wait_queue(&dev->r_wait, &wait); <span class="comment">/* 将对应的队列项从等待队列头删除 */</span></span><br><span class="line"> <span class="keyword">return</span> ret;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/* 操作集 */</span></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">file_operations</span> <span class="title">keyirq_fops</span> = {</span></span><br><span class="line"> .owner = THIS_MODULE,</span><br><span class="line"> .<span class="built_in">open</span> = keyirq_open,</span><br><span class="line"> .<span class="built_in">release</span> = keyirq_release,</span><br><span class="line"> .<span class="built_in">write</span> = keyirq_write,</span><br><span class="line"> .<span class="built_in">read</span> = keyirq_read,</span><br><span class="line">};</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 定时器处理函数</span></span><br><span class="line"><span class="comment"> * @param arg 用户参数</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">void</span> <span class="title">timer_func</span><span class="params">(<span class="keyword">unsigned</span> <span class="keyword">long</span> arg)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = (<span class="title">struct</span> <span class="title">keyirq_dev</span>*)<span class="title">arg</span>;</span></span><br><span class="line"> <span class="keyword">int</span> value = <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line"> value = gpio_get_value(dev->key[<span class="number">0</span>].gpio);</span><br><span class="line"> <span class="keyword">if</span> (value == <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">// 按键按下</span></span><br><span class="line"> printk(<span class="string">"KEY0 Press!\r\n"</span>);</span><br><span class="line"> atomic_set(&dev->keyValue, dev->key[<span class="number">0</span>].value);</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span> <span class="keyword">if</span>(value == <span class="number">1</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">// 按键释放</span></span><br><span class="line"> printk(<span class="string">"KEY0 Release!\r\n"</span>);</span><br><span class="line"> atomic_set(&dev->keyValue, dev->key[<span class="number">0</span>].value | <span class="number">0X80</span>); <span class="comment">// 打上标签,标记按键按下</span></span><br><span class="line"> atomic_set(&dev->keyRelease, <span class="number">1</span>);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 唤醒进程 */</span></span><br><span class="line"> <span class="keyword">if</span>(atomic_read(&dev->keyRelease))</span><br><span class="line"> {</span><br><span class="line"> wake_up(&dev->r_wait);</span><br><span class="line"> }</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief key0 tasklet 处理函数</span></span><br><span class="line"><span class="comment"> * @param arg 传递的参数</span></span><br><span class="line"><span class="comment"> */</span> </span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">void</span> <span class="title">key0_tasklet</span><span class="params">(<span class="keyword">unsigned</span> <span class="keyword">long</span> arg)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = (<span class="title">struct</span> <span class="title">keyirq_dev</span> *)<span class="title">arg</span>;</span></span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 启动定时器来延时消抖 */</span></span><br><span class="line"> dev->timer.data = (<span class="keyword">volatile</span> <span class="keyword">unsigned</span> <span class="keyword">long</span>)arg;</span><br><span class="line"> mod_timer(&dev->timer, jiffies + msecs_to_jiffies(<span class="number">20</span>));</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 按键中断处理函数</span></span><br><span class="line"><span class="comment"> * @param irq 中断号</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">irqreturn_t</span> <span class="title">key0_handler</span><span class="params">(<span class="keyword">int</span> irq, <span class="keyword">void</span> *arg)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">arg</span>;</span></span><br><span class="line"></span><br><span class="line"> tasklet_schedule(&dev->key[<span class="number">0</span>].tasklet); <span class="comment">/* 调度对应的tasklet */</span></span><br><span class="line"> <span class="keyword">return</span> IRQ_HANDLED;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 按键初始化</span></span><br><span class="line"><span class="comment"> * @param dev 设备结构体</span></span><br><span class="line"><span class="comment"> * @return 错误类型</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> <span class="title">keyio_init</span><span class="params">(struct keyirq_dev *dev)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> ret = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">int</span> i = <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 按键初始化 */</span></span><br><span class="line"> dev->nd = of_find_node_by_path(<span class="string">"/key"</span>);</span><br><span class="line"> <span class="keyword">if</span> (dev->nd == <span class="literal">NULL</span>)</span><br><span class="line"> {</span><br><span class="line"> ret = -EINVAL;</span><br><span class="line"> <span class="keyword">goto</span> fail_nd;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 可能有多个按键,因此需要根据实际数量来读取 */</span></span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> dev->key[i].gpio = of_get_named_gpio(dev->nd, <span class="string">"key-gpios"</span>, i);</span><br><span class="line"> <span class="keyword">if</span> (dev->key[i].gpio < <span class="number">0</span> )</span><br><span class="line"> {</span><br><span class="line"> ret = -EINVAL;</span><br><span class="line"> <span class="keyword">goto</span> fail_getGpio;</span><br><span class="line"> }</span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> <span class="built_in">memset</span>(dev->key[i].name, <span class="number">0</span>, <span class="keyword">sizeof</span>(dev->key[i].name));</span><br><span class="line"> <span class="built_in">sprintf</span>(dev->key[i].name, <span class="string">"KEY%d"</span>, i);</span><br><span class="line"> ret = gpio_request(dev->key[i].gpio, dev->key[i].name);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span> )</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_requestGpio;</span><br><span class="line"> }</span><br><span class="line"> ret = gpio_direction_input(dev->key[i].gpio);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_setGpioDir;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 获取中断号,两种方式皆可 */</span></span><br><span class="line"> dev->key[i].irqNum = gpio_to_irq(dev->key[i].gpio);</span><br><span class="line"> <span class="comment">// dev->key[i].irqNum = irq_of_parse_and_map(dev->nd, i);</span></span><br><span class="line"></span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 按键中断初始化 */</span></span><br><span class="line"> dev->key[<span class="number">0</span>].handler = key0_handler;</span><br><span class="line"> dev->key[<span class="number">0</span>].value = KEY0_VALUE;</span><br><span class="line"></span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 跳变沿触发方式 */</span></span><br><span class="line"> ret = request_irq(dev->key[i].irqNum, dev->key[i].handler, IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING, dev->key[i].name, dev);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> printk(<span class="string">"irq %d request failed!\r\n"</span>, dev->key[i].irqNum);</span><br><span class="line"> <span class="keyword">goto</span> fail_irq;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* <span class="doctag">NOTE:</span>没有根据多个按键来初始化各自的tasklet,实际中需要根据情况来编写 */</span></span><br><span class="line"> tasklet_init(&dev->key[i].tasklet, key0_tasklet, (<span class="keyword">unsigned</span> <span class="keyword">long</span>)dev);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化定时器 */</span></span><br><span class="line"> init_timer(&dev->timer);</span><br><span class="line"> dev->timer.function = timer_func;</span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line">fail_irq:</span><br><span class="line">fail_setGpioDir:</span><br><span class="line"> <span class="keyword">for</span> (i = i<span class="number">-1</span>; i >= <span class="number">0</span>; i--)</span><br><span class="line"> {</span><br><span class="line"> gpio_free(dev->key[i].gpio);</span><br><span class="line"> }</span><br><span class="line">fail_requestGpio:</span><br><span class="line">fail_getGpio:</span><br><span class="line">fail_nd:</span><br><span class="line"> <span class="keyword">return</span> ret;</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment">/* 入口和出口 */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> __init <span class="title">keyirq_init</span><span class="params">(<span class="keyword">void</span>)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> ret = <span class="number">0</span>;</span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 注册字符设备ID */</span></span><br><span class="line"> keyirq.major = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">if</span>(keyirq.major)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 给定设备号 */</span></span><br><span class="line"> keyirq.devid = MKDEV(keyirq.major, <span class="number">0</span>);</span><br><span class="line"> ret = register_chrdev_region(keyirq.devid, KEYIRQ_CNT, KEYIRQ_NAME);</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> alloc_chrdev_region(&keyirq.devid, <span class="number">0</span>,KEYIRQ_CNT,KEYIRQ_NAME);</span><br><span class="line"> keyirq.major = MAJOR(keyirq.devid);</span><br><span class="line"> keyirq.minor = MINOR(keyirq.devid);</span><br><span class="line"> printk(<span class="string">"dev Major ID:%d\r\n"</span>,keyirq.major);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_devid;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化字符设备 */</span></span><br><span class="line"> keyirq.cdev.owner = THIS_MODULE;</span><br><span class="line"> cdev_init(&keyirq.cdev, &keyirq_fops);</span><br><span class="line"> ret = cdev_add(&keyirq.cdev, keyirq.devid, KEYIRQ_CNT);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_cdev;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 创建设备类 */</span></span><br><span class="line"> keyirq<span class="class">.<span class="keyword">class</span> = <span class="title">class_create</span>(<span class="title">THIS_MODULE</span>, <span class="title">KEYIRQ_NAME</span>);</span></span><br><span class="line"> <span class="keyword">if</span> (IS_ERR(keyirq.class))</span><br><span class="line"> {</span><br><span class="line"> ret = PTR_ERR(keyirq.class);</span><br><span class="line"> <span class="keyword">goto</span> fail_class;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 创建设备节点 */</span></span><br><span class="line"> keyirq.device = device_create(keyirq.class, <span class="literal">NULL</span>, keyirq.devid, <span class="literal">NULL</span>, KEYIRQ_NAME);</span><br><span class="line"> <span class="keyword">if</span> (IS_ERR(keyirq.device))</span><br><span class="line"> {</span><br><span class="line"> ret = PTR_ERR(keyirq.device);</span><br><span class="line"> <span class="keyword">goto</span> fail_device;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化IO */</span></span><br><span class="line"> ret = keyio_init(&keyirq);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_init;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化原子变量 */</span></span><br><span class="line"> atomic_set(&keyirq.keyValue, INVAKEY);</span><br><span class="line"> atomic_set(&keyirq.keyRelease, <span class="number">0</span>);</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化等待队列头 */</span></span><br><span class="line"> init_waitqueue_head(&keyirq.r_wait);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line">fail_init:</span><br><span class="line"> device_destroy(keyirq.class, keyirq.devid);</span><br><span class="line">fail_device:</span><br><span class="line"> class_destroy(keyirq.class);</span><br><span class="line">fail_class:</span><br><span class="line"> cdev_del(&keyirq.cdev);</span><br><span class="line">fail_cdev:</span><br><span class="line"> unregister_chrdev_region(keyirq.devid, KEYIRQ_CNT);</span><br><span class="line">fail_devid:</span><br><span class="line"> <span class="keyword">return</span> ret;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">void</span> __exit <span class="title">keyirq_exit</span><span class="params">(<span class="keyword">void</span>)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> i = <span class="number">0</span>;</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 释放中断和IO */</span></span><br><span class="line"> free_irq(keyirq.key[i].irqNum, &keyirq);</span><br><span class="line"> gpio_free(keyirq.key[i].gpio);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 删除定时器 */</span></span><br><span class="line"> del_timer_sync(&keyirq.timer);</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 注销字符设备驱动 */</span></span><br><span class="line"> device_destroy(keyirq.class, keyirq.devid);</span><br><span class="line"> class_destroy(keyirq.class);</span><br><span class="line"> cdev_del(&keyirq.cdev);</span><br><span class="line"> unregister_chrdev_region(keyirq.devid,KEYIRQ_CNT);</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/* 注册驱动和卸载驱动 */</span></span><br><span class="line">module_init(keyirq_init);</span><br><span class="line">module_exit(keyirq_exit);</span><br><span class="line"></span><br><span class="line">MODULE_LICENSE(<span class="string">"GPL"</span>);</span><br><span class="line">MODULE_AUTHOR(<span class="string">"fengyuhang"</span>);</span><br></pre></td></tr></table></figure><h3 id="2-2-非阻塞式访问"><a href="#2-2-非阻塞式访问" class="headerlink" title="2.2 非阻塞式访问"></a>2.2 非阻塞式访问</h3><ol><li>驱动程序</li></ol><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br><span class="line">39</span><br><span class="line">40</span><br><span class="line">41</span><br><span class="line">42</span><br><span class="line">43</span><br><span class="line">44</span><br><span class="line">45</span><br><span class="line">46</span><br><span class="line">47</span><br><span class="line">48</span><br><span class="line">49</span><br><span class="line">50</span><br><span class="line">51</span><br><span class="line">52</span><br><span class="line">53</span><br><span class="line">54</span><br><span class="line">55</span><br><span class="line">56</span><br><span class="line">57</span><br><span class="line">58</span><br><span class="line">59</span><br><span class="line">60</span><br><span class="line">61</span><br><span class="line">62</span><br><span class="line">63</span><br><span class="line">64</span><br><span class="line">65</span><br><span class="line">66</span><br><span class="line">67</span><br><span class="line">68</span><br><span class="line">69</span><br><span class="line">70</span><br><span class="line">71</span><br><span class="line">72</span><br><span class="line">73</span><br><span class="line">74</span><br><span class="line">75</span><br><span class="line">76</span><br><span class="line">77</span><br><span class="line">78</span><br><span class="line">79</span><br><span class="line">80</span><br><span class="line">81</span><br><span class="line">82</span><br><span class="line">83</span><br><span class="line">84</span><br><span class="line">85</span><br><span class="line">86</span><br><span class="line">87</span><br><span class="line">88</span><br><span class="line">89</span><br><span class="line">90</span><br><span class="line">91</span><br><span class="line">92</span><br><span class="line">93</span><br><span class="line">94</span><br><span class="line">95</span><br><span class="line">96</span><br><span class="line">97</span><br><span class="line">98</span><br><span class="line">99</span><br><span class="line">100</span><br><span class="line">101</span><br><span class="line">102</span><br><span class="line">103</span><br><span class="line">104</span><br><span class="line">105</span><br><span class="line">106</span><br><span class="line">107</span><br><span class="line">108</span><br><span class="line">109</span><br><span class="line">110</span><br><span class="line">111</span><br><span class="line">112</span><br><span class="line">113</span><br><span class="line">114</span><br><span class="line">115</span><br><span class="line">116</span><br><span class="line">117</span><br><span class="line">118</span><br><span class="line">119</span><br><span class="line">120</span><br><span class="line">121</span><br><span class="line">122</span><br><span class="line">123</span><br><span class="line">124</span><br><span class="line">125</span><br><span class="line">126</span><br><span class="line">127</span><br><span class="line">128</span><br><span class="line">129</span><br><span class="line">130</span><br><span class="line">131</span><br><span class="line">132</span><br><span class="line">133</span><br><span class="line">134</span><br><span class="line">135</span><br><span class="line">136</span><br><span class="line">137</span><br><span class="line">138</span><br><span class="line">139</span><br><span class="line">140</span><br><span class="line">141</span><br><span class="line">142</span><br><span class="line">143</span><br><span class="line">144</span><br><span class="line">145</span><br><span class="line">146</span><br><span class="line">147</span><br><span class="line">148</span><br><span class="line">149</span><br><span class="line">150</span><br><span class="line">151</span><br><span class="line">152</span><br><span class="line">153</span><br><span class="line">154</span><br><span class="line">155</span><br><span class="line">156</span><br><span class="line">157</span><br><span class="line">158</span><br><span class="line">159</span><br><span class="line">160</span><br><span class="line">161</span><br><span class="line">162</span><br><span class="line">163</span><br><span class="line">164</span><br><span class="line">165</span><br><span class="line">166</span><br><span class="line">167</span><br><span class="line">168</span><br><span class="line">169</span><br><span class="line">170</span><br><span class="line">171</span><br><span class="line">172</span><br><span class="line">173</span><br><span class="line">174</span><br><span class="line">175</span><br><span class="line">176</span><br><span class="line">177</span><br><span class="line">178</span><br><span class="line">179</span><br><span class="line">180</span><br><span class="line">181</span><br><span class="line">182</span><br><span class="line">183</span><br><span class="line">184</span><br><span class="line">185</span><br><span class="line">186</span><br><span class="line">187</span><br><span class="line">188</span><br><span class="line">189</span><br><span class="line">190</span><br><span class="line">191</span><br><span class="line">192</span><br><span class="line">193</span><br><span class="line">194</span><br><span class="line">195</span><br><span class="line">196</span><br><span class="line">197</span><br><span class="line">198</span><br><span class="line">199</span><br><span class="line">200</span><br><span class="line">201</span><br><span class="line">202</span><br><span class="line">203</span><br><span class="line">204</span><br><span class="line">205</span><br><span class="line">206</span><br><span class="line">207</span><br><span class="line">208</span><br><span class="line">209</span><br><span class="line">210</span><br><span class="line">211</span><br><span class="line">212</span><br><span class="line">213</span><br><span class="line">214</span><br><span class="line">215</span><br><span class="line">216</span><br><span class="line">217</span><br><span class="line">218</span><br><span class="line">219</span><br><span class="line">220</span><br><span class="line">221</span><br><span class="line">222</span><br><span class="line">223</span><br><span class="line">224</span><br><span class="line">225</span><br><span class="line">226</span><br><span class="line">227</span><br><span class="line">228</span><br><span class="line">229</span><br><span class="line">230</span><br><span class="line">231</span><br><span class="line">232</span><br><span class="line">233</span><br><span class="line">234</span><br><span class="line">235</span><br><span class="line">236</span><br><span class="line">237</span><br><span class="line">238</span><br><span class="line">239</span><br><span class="line">240</span><br><span class="line">241</span><br><span class="line">242</span><br><span class="line">243</span><br><span class="line">244</span><br><span class="line">245</span><br><span class="line">246</span><br><span class="line">247</span><br><span class="line">248</span><br><span class="line">249</span><br><span class="line">250</span><br><span class="line">251</span><br><span class="line">252</span><br><span class="line">253</span><br><span class="line">254</span><br><span class="line">255</span><br><span class="line">256</span><br><span class="line">257</span><br><span class="line">258</span><br><span class="line">259</span><br><span class="line">260</span><br><span class="line">261</span><br><span class="line">262</span><br><span class="line">263</span><br><span class="line">264</span><br><span class="line">265</span><br><span class="line">266</span><br><span class="line">267</span><br><span class="line">268</span><br><span class="line">269</span><br><span class="line">270</span><br><span class="line">271</span><br><span class="line">272</span><br><span class="line">273</span><br><span class="line">274</span><br><span class="line">275</span><br><span class="line">276</span><br><span class="line">277</span><br><span class="line">278</span><br><span class="line">279</span><br><span class="line">280</span><br><span class="line">281</span><br><span class="line">282</span><br><span class="line">283</span><br><span class="line">284</span><br><span class="line">285</span><br><span class="line">286</span><br><span class="line">287</span><br><span class="line">288</span><br><span class="line">289</span><br><span class="line">290</span><br><span class="line">291</span><br><span class="line">292</span><br><span class="line">293</span><br><span class="line">294</span><br><span class="line">295</span><br><span class="line">296</span><br><span class="line">297</span><br><span class="line">298</span><br><span class="line">299</span><br><span class="line">300</span><br><span class="line">301</span><br><span class="line">302</span><br><span class="line">303</span><br><span class="line">304</span><br><span class="line">305</span><br><span class="line">306</span><br><span class="line">307</span><br><span class="line">308</span><br><span class="line">309</span><br><span class="line">310</span><br><span class="line">311</span><br><span class="line">312</span><br><span class="line">313</span><br><span class="line">314</span><br><span class="line">315</span><br><span class="line">316</span><br><span class="line">317</span><br><span class="line">318</span><br><span class="line">319</span><br><span class="line">320</span><br><span class="line">321</span><br><span class="line">322</span><br><span class="line">323</span><br><span class="line">324</span><br><span class="line">325</span><br><span class="line">326</span><br><span class="line">327</span><br><span class="line">328</span><br><span class="line">329</span><br><span class="line">330</span><br><span class="line">331</span><br><span class="line">332</span><br><span class="line">333</span><br><span class="line">334</span><br><span class="line">335</span><br><span class="line">336</span><br><span class="line">337</span><br><span class="line">338</span><br><span class="line">339</span><br><span class="line">340</span><br><span class="line">341</span><br><span class="line">342</span><br><span class="line">343</span><br><span class="line">344</span><br><span class="line">345</span><br><span class="line">346</span><br><span class="line">347</span><br><span class="line">348</span><br><span class="line">349</span><br><span class="line">350</span><br><span class="line">351</span><br><span class="line">352</span><br><span class="line">353</span><br><span class="line">354</span><br><span class="line">355</span><br><span class="line">356</span><br><span class="line">357</span><br><span class="line">358</span><br><span class="line">359</span><br><span class="line">360</span><br><span class="line">361</span><br><span class="line">362</span><br><span class="line">363</span><br><span class="line">364</span><br><span class="line">365</span><br><span class="line">366</span><br><span class="line">367</span><br><span class="line">368</span><br><span class="line">369</span><br><span class="line">370</span><br><span class="line">371</span><br><span class="line">372</span><br><span class="line">373</span><br><span class="line">374</span><br><span class="line">375</span><br><span class="line">376</span><br><span class="line">377</span><br><span class="line">378</span><br><span class="line">379</span><br><span class="line">380</span><br><span class="line">381</span><br><span class="line">382</span><br><span class="line">383</span><br><span class="line">384</span><br><span class="line">385</span><br><span class="line">386</span><br><span class="line">387</span><br><span class="line">388</span><br><span class="line">389</span><br><span class="line">390</span><br><span class="line">391</span><br><span class="line">392</span><br><span class="line">393</span><br><span class="line">394</span><br><span class="line">395</span><br><span class="line">396</span><br><span class="line">397</span><br><span class="line">398</span><br><span class="line">399</span><br><span class="line">400</span><br><span class="line">401</span><br><span class="line">402</span><br><span class="line">403</span><br><span class="line">404</span><br><span class="line">405</span><br><span class="line">406</span><br><span class="line">407</span><br><span class="line">408</span><br><span class="line">409</span><br><span class="line">410</span><br><span class="line">411</span><br><span class="line">412</span><br><span class="line">413</span><br><span class="line">414</span><br><span class="line">415</span><br><span class="line">416</span><br><span class="line">417</span><br><span class="line">418</span><br><span class="line">419</span><br><span class="line">420</span><br><span class="line">421</span><br><span class="line">422</span><br><span class="line">423</span><br><span class="line">424</span><br><span class="line">425</span><br><span class="line">426</span><br><span class="line">427</span><br><span class="line">428</span><br><span class="line">429</span><br><span class="line">430</span><br><span class="line">431</span><br><span class="line">432</span><br></pre></td><td class="code"><pre><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/types.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/kernel.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/delay.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/init.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/ide.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/module.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/errno.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/gpio.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/cdev.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of_address.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of_irq.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/of_gpio.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/timer.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/jiffies.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/string.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/irq.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/interrupt.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><linux/poll.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><asm/mach/map.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><asm/uaccess.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><asm/io.h></span></span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEYIRQ_CNT 1</span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEYIRQ_NAME <span class="meta-string">"keyirq"</span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEY_NUM 1</span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> KEY0_VALUE 0X01</span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">define</span> INVAKEY 0XFF <span class="comment">/* 无效按键值 */</span></span></span><br><span class="line"></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">key_dev</span></span></span><br><span class="line"><span class="class">{</span></span><br><span class="line"> <span class="keyword">int</span> gpio; <span class="comment">/* IO编号 */</span></span><br><span class="line"> <span class="keyword">int</span> irqNum; <span class="comment">/* 中断号 */</span></span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">char</span> value; <span class="comment">/* 按键值 */</span></span><br><span class="line"> <span class="keyword">char</span> name[<span class="number">10</span>]; <span class="comment">/* 名称 */</span></span><br><span class="line"> <span class="keyword">irqreturn_t</span> (*handler)(<span class="keyword">int</span>, <span class="keyword">void</span>*); <span class="comment">/* 中断处理函数 */</span></span><br><span class="line"></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">tasklet_struct</span> <span class="title">tasklet</span>;</span></span><br><span class="line">};</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment">/* 定义设备驱动结构体 */</span></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span></span></span><br><span class="line"><span class="class">{</span></span><br><span class="line"> <span class="keyword">dev_t</span> devid; <span class="comment">/* 设备ID */</span></span><br><span class="line"> <span class="keyword">int</span> major; <span class="comment">/* 主设备号 */</span></span><br><span class="line"> <span class="keyword">int</span> minor; <span class="comment">/* 次设备号 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">cdev</span> <span class="title">cdev</span>;</span> <span class="comment">/* 字符设备结构体 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">class</span> *<span class="title">class</span>;</span> <span class="comment">/* 设备类 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">device</span> *<span class="title">device</span>;</span> <span class="comment">/* 设备节点 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">device_node</span> *<span class="title">nd</span>;</span> <span class="comment">/* 设备树节点 */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">key_dev</span> <span class="title">key</span>[<span class="title">KEY_NUM</span>];</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">timer_list</span> <span class="title">timer</span>;</span> <span class="comment">/* 定时器 */</span></span><br><span class="line"></span><br><span class="line"> <span class="keyword">atomic_t</span> keyValue; <span class="comment">/* 按键值 */</span></span><br><span class="line"> <span class="keyword">atomic_t</span> keyRelease; <span class="comment">/* 按键是否释放 */</span></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">wait_queue_head_t</span> r_wait; <span class="comment">/* 读等待队列头 */</span></span><br><span class="line">};</span><br><span class="line"></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> <span class="title">keyirq</span>;</span> <span class="comment">/* 定义一个keyirq设备 */</span></span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> <span class="title">keyirq_open</span><span class="params">(struct inode *inode, struct file *filp)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> filp->private_data = &keyirq;</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line">} </span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> <span class="title">keyirq_release</span><span class="params">(struct inode *inode, struct file *filp)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">filp</span>-><span class="title">private_data</span>;</span></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">ssize_t</span> <span class="title">keyirq_write</span><span class="params">(struct file *filp, <span class="keyword">const</span> <span class="keyword">char</span> __user *buf, <span class="keyword">size_t</span> count, <span class="keyword">loff_t</span> *ppos)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">filp</span>-><span class="title">private_data</span>;</span></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">ssize_t</span> <span class="title">keyirq_read</span><span class="params">(struct file *filp, <span class="keyword">char</span> __user *buf, <span class="keyword">size_t</span> count, <span class="keyword">loff_t</span> *ppos)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">filp</span>-><span class="title">private_data</span>;</span></span><br><span class="line"> <span class="keyword">int</span> ret = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">char</span> keyValue;</span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">char</span> keyRelease;</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span>(filp->f_flags & O_NONBLOCK) </span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 非阻塞方式访问 */</span></span><br><span class="line"> <span class="keyword">if</span>(atomic_read(&dev->keyRelease) == <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">return</span> -EAGAIN;</span><br><span class="line"> }</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 阻塞访问方式 */</span></span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 等待事件 */</span></span><br><span class="line"> wait_event_interruptible(dev->r_wait, atomic_read(&dev->keyRelease));<span class="comment">/* 等待按键值有效 */</span></span><br><span class="line"></span><br><span class="line"> <span class="comment">// <span class="doctag">NOTE:</span>wait_event 不可以被信号打断,使用 kill -9 PID 无法杀掉任务</span></span><br><span class="line"> <span class="comment">// wait_event(dev->r_wait, atomic_read(&dev->keyRelease)); /* 等待按键值有效 */</span></span><br><span class="line"></span><br><span class="line"> <span class="meta">#<span class="meta-keyword">if</span> 0</span></span><br><span class="line"> DECLARE_WAITQUEUE(wait, current); <span class="comment">/* 定义一个等待队列项 */</span></span><br><span class="line"> add_wait_queue(&dev->r_wait, &wait); <span class="comment">/* 将等待队列项添加到等待队列头中 */</span></span><br><span class="line"> __set_current_state(TASK_INTERRUPTIBLE); <span class="comment">/* 设置当前进程为可以被打断的状态 */</span></span><br><span class="line"> schedule(); <span class="comment">/* 切换 */</span></span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 唤醒后,进程从这里运行 */</span></span><br><span class="line"> <span class="keyword">if</span> (signal_pending(current)) <span class="comment">/* 判断当前进程是否有信号需要处理,返回值不为零表示有信号需要进行处理 */</span></span><br><span class="line"> {</span><br><span class="line"> ret = -ERESTARTSYS;</span><br><span class="line"> <span class="keyword">goto</span> data_err;</span><br><span class="line"> }</span><br><span class="line"> <span class="meta">#<span class="meta-keyword">endif</span></span></span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> keyValue = atomic_read(&dev->keyValue);</span><br><span class="line"> keyRelease = atomic_read(&dev->keyRelease);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (keyRelease)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 一次有效按键 */</span></span><br><span class="line"> <span class="keyword">if</span> (keyValue & <span class="number">0X80</span>)</span><br><span class="line"> {</span><br><span class="line"> keyValue &= ~<span class="number">0X80</span>; <span class="comment">// 去除标记</span></span><br><span class="line"> ret = copy_to_user(buf, &keyValue, <span class="keyword">sizeof</span>(keyValue));</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> data_err;</span><br><span class="line"> }</span><br><span class="line"> atomic_set(&dev->keyRelease, <span class="number">0</span>);</span><br><span class="line"> } </span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> data_err;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> </span><br><span class="line">data_err:</span><br><span class="line"> <span class="meta">#<span class="meta-keyword">if</span> 0</span></span><br><span class="line"> __set_current_state(TASK_RUNNING); <span class="comment">/* 设置当前任务为运行状态 */</span></span><br><span class="line"> remove_wait_queue(&dev->r_wait, &wait); <span class="comment">/* 将对应的队列项从等待队列头删除 */</span></span><br><span class="line"> <span class="meta">#<span class="meta-keyword">endif</span></span></span><br><span class="line"> <span class="keyword">return</span> ret;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">unsigned</span> <span class="keyword">int</span> <span class="title">keyirq_poll</span><span class="params">(struct file *filp, struct poll_table_struct *wait)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> mask = <span class="number">0</span>;</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">filp</span>-><span class="title">private_data</span>;</span></span><br><span class="line"></span><br><span class="line"> poll_wait(filp, &dev->r_wait, wait);</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 是否可读 */</span></span><br><span class="line"> <span class="keyword">if</span> (atomic_read(&dev->keyRelease))</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 按键按下,可读 */</span></span><br><span class="line"> mask = POLL_IN | POLLRDNORM; <span class="comment">/* 返回POLL_IN */</span></span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="keyword">return</span> mask;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/* 操作集 */</span></span><br><span class="line"><span class="class"><span class="keyword">struct</span> <span class="title">file_operations</span> <span class="title">keyirq_fops</span> = {</span></span><br><span class="line"> .owner = THIS_MODULE,</span><br><span class="line"> .<span class="built_in">open</span> = keyirq_open,</span><br><span class="line"> .<span class="built_in">release</span> = keyirq_release,</span><br><span class="line"> .<span class="built_in">write</span> = keyirq_write,</span><br><span class="line"> .<span class="built_in">read</span> = keyirq_read,</span><br><span class="line"> .poll = keyirq_poll, </span><br><span class="line">};</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 定时器处理函数</span></span><br><span class="line"><span class="comment"> * @param arg 用户参数</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">void</span> <span class="title">timer_func</span><span class="params">(<span class="keyword">unsigned</span> <span class="keyword">long</span> arg)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = (<span class="title">struct</span> <span class="title">keyirq_dev</span>*)<span class="title">arg</span>;</span></span><br><span class="line"> <span class="keyword">int</span> value = <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line"> value = gpio_get_value(dev->key[<span class="number">0</span>].gpio);</span><br><span class="line"> <span class="keyword">if</span> (value == <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">// 按键按下</span></span><br><span class="line"> printk(<span class="string">"KEY0 Press!\r\n"</span>);</span><br><span class="line"> atomic_set(&dev->keyValue, dev->key[<span class="number">0</span>].value);</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span> <span class="keyword">if</span>(value == <span class="number">1</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">// 按键释放</span></span><br><span class="line"> printk(<span class="string">"KEY0 Release!\r\n"</span>);</span><br><span class="line"> atomic_set(&dev->keyValue, dev->key[<span class="number">0</span>].value | <span class="number">0X80</span>); <span class="comment">// 打上标签,标记按键按下</span></span><br><span class="line"> atomic_set(&dev->keyRelease, <span class="number">1</span>);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 唤醒进程 */</span></span><br><span class="line"> <span class="keyword">if</span>(atomic_read(&dev->keyRelease))</span><br><span class="line"> {</span><br><span class="line"> wake_up(&dev->r_wait);</span><br><span class="line"> }</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief key0 tasklet 处理函数</span></span><br><span class="line"><span class="comment"> * @param arg 传递的参数</span></span><br><span class="line"><span class="comment"> */</span> </span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">void</span> <span class="title">key0_tasklet</span><span class="params">(<span class="keyword">unsigned</span> <span class="keyword">long</span> arg)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = (<span class="title">struct</span> <span class="title">keyirq_dev</span> *)<span class="title">arg</span>;</span></span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 启动定时器来延时消抖 */</span></span><br><span class="line"> dev->timer.data = (<span class="keyword">volatile</span> <span class="keyword">unsigned</span> <span class="keyword">long</span>)arg;</span><br><span class="line"> mod_timer(&dev->timer, jiffies + msecs_to_jiffies(<span class="number">20</span>));</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 按键中断处理函数</span></span><br><span class="line"><span class="comment"> * @param irq 中断号</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">irqreturn_t</span> <span class="title">key0_handler</span><span class="params">(<span class="keyword">int</span> irq, <span class="keyword">void</span> *arg)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">keyirq_dev</span> *<span class="title">dev</span> = <span class="title">arg</span>;</span></span><br><span class="line"></span><br><span class="line"> tasklet_schedule(&dev->key[<span class="number">0</span>].tasklet); <span class="comment">/* 调度对应的tasklet */</span></span><br><span class="line"> <span class="keyword">return</span> IRQ_HANDLED;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * @brief 按键初始化</span></span><br><span class="line"><span class="comment"> * @param dev 设备结构体</span></span><br><span class="line"><span class="comment"> * @return 错误类型</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> <span class="title">keyio_init</span><span class="params">(struct keyirq_dev *dev)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> ret = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">int</span> i = <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 按键初始化 */</span></span><br><span class="line"> dev->nd = of_find_node_by_path(<span class="string">"/key"</span>);</span><br><span class="line"> <span class="keyword">if</span> (dev->nd == <span class="literal">NULL</span>)</span><br><span class="line"> {</span><br><span class="line"> ret = -EINVAL;</span><br><span class="line"> <span class="keyword">goto</span> fail_nd;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 可能有多个按键,因此需要根据实际数量来读取 */</span></span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> dev->key[i].gpio = of_get_named_gpio(dev->nd, <span class="string">"key-gpios"</span>, i);</span><br><span class="line"> <span class="keyword">if</span> (dev->key[i].gpio < <span class="number">0</span> )</span><br><span class="line"> {</span><br><span class="line"> ret = -EINVAL;</span><br><span class="line"> <span class="keyword">goto</span> fail_getGpio;</span><br><span class="line"> }</span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> <span class="built_in">memset</span>(dev->key[i].name, <span class="number">0</span>, <span class="keyword">sizeof</span>(dev->key[i].name));</span><br><span class="line"> <span class="built_in">sprintf</span>(dev->key[i].name, <span class="string">"KEY%d"</span>, i);</span><br><span class="line"> ret = gpio_request(dev->key[i].gpio, dev->key[i].name);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span> )</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_requestGpio;</span><br><span class="line"> }</span><br><span class="line"> ret = gpio_direction_input(dev->key[i].gpio);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_setGpioDir;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 获取中断号,两种方式皆可 */</span></span><br><span class="line"> dev->key[i].irqNum = gpio_to_irq(dev->key[i].gpio);</span><br><span class="line"> <span class="comment">// dev->key[i].irqNum = irq_of_parse_and_map(dev->nd, i);</span></span><br><span class="line"></span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 按键中断初始化 */</span></span><br><span class="line"> dev->key[<span class="number">0</span>].handler = key0_handler;</span><br><span class="line"> dev->key[<span class="number">0</span>].value = KEY0_VALUE;</span><br><span class="line"></span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 跳变沿触发方式 */</span></span><br><span class="line"> ret = request_irq(dev->key[i].irqNum, dev->key[i].handler, IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING, dev->key[i].name, dev);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> printk(<span class="string">"irq %d request failed!\r\n"</span>, dev->key[i].irqNum);</span><br><span class="line"> <span class="keyword">goto</span> fail_irq;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* <span class="doctag">NOTE:</span>没有根据多个按键来初始化各自的tasklet,实际中需要根据情况来编写 */</span></span><br><span class="line"> tasklet_init(&dev->key[i].tasklet, key0_tasklet, (<span class="keyword">unsigned</span> <span class="keyword">long</span>)dev);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化定时器 */</span></span><br><span class="line"> init_timer(&dev->timer);</span><br><span class="line"> dev->timer.function = timer_func;</span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line">fail_irq:</span><br><span class="line">fail_setGpioDir:</span><br><span class="line"> <span class="keyword">for</span> (i = i<span class="number">-1</span>; i >= <span class="number">0</span>; i--)</span><br><span class="line"> {</span><br><span class="line"> gpio_free(dev->key[i].gpio);</span><br><span class="line"> }</span><br><span class="line">fail_requestGpio:</span><br><span class="line">fail_getGpio:</span><br><span class="line">fail_nd:</span><br><span class="line"> <span class="keyword">return</span> ret;</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment">/* 入口和出口 */</span></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">int</span> __init <span class="title">keyirq_init</span><span class="params">(<span class="keyword">void</span>)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> ret = <span class="number">0</span>;</span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 注册字符设备ID */</span></span><br><span class="line"> keyirq.major = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">if</span>(keyirq.major)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 给定设备号 */</span></span><br><span class="line"> keyirq.devid = MKDEV(keyirq.major, <span class="number">0</span>);</span><br><span class="line"> ret = register_chrdev_region(keyirq.devid, KEYIRQ_CNT, KEYIRQ_NAME);</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> alloc_chrdev_region(&keyirq.devid, <span class="number">0</span>,KEYIRQ_CNT,KEYIRQ_NAME);</span><br><span class="line"> keyirq.major = MAJOR(keyirq.devid);</span><br><span class="line"> keyirq.minor = MINOR(keyirq.devid);</span><br><span class="line"> printk(<span class="string">"dev Major ID:%d\r\n"</span>,keyirq.major);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_devid;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化字符设备 */</span></span><br><span class="line"> keyirq.cdev.owner = THIS_MODULE;</span><br><span class="line"> cdev_init(&keyirq.cdev, &keyirq_fops);</span><br><span class="line"> ret = cdev_add(&keyirq.cdev, keyirq.devid, KEYIRQ_CNT);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_cdev;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 创建设备类 */</span></span><br><span class="line"> keyirq<span class="class">.<span class="keyword">class</span> = <span class="title">class_create</span>(<span class="title">THIS_MODULE</span>, <span class="title">KEYIRQ_NAME</span>);</span></span><br><span class="line"> <span class="keyword">if</span> (IS_ERR(keyirq.class))</span><br><span class="line"> {</span><br><span class="line"> ret = PTR_ERR(keyirq.class);</span><br><span class="line"> <span class="keyword">goto</span> fail_class;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 创建设备节点 */</span></span><br><span class="line"> keyirq.device = device_create(keyirq.class, <span class="literal">NULL</span>, keyirq.devid, <span class="literal">NULL</span>, KEYIRQ_NAME);</span><br><span class="line"> <span class="keyword">if</span> (IS_ERR(keyirq.device))</span><br><span class="line"> {</span><br><span class="line"> ret = PTR_ERR(keyirq.device);</span><br><span class="line"> <span class="keyword">goto</span> fail_device;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化IO */</span></span><br><span class="line"> ret = keyio_init(&keyirq);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">goto</span> fail_init;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化原子变量 */</span></span><br><span class="line"> atomic_set(&keyirq.keyValue, INVAKEY);</span><br><span class="line"> atomic_set(&keyirq.keyRelease, <span class="number">0</span>);</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 初始化等待队列头 */</span></span><br><span class="line"> init_waitqueue_head(&keyirq.r_wait);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line">fail_init:</span><br><span class="line"> device_destroy(keyirq.class, keyirq.devid);</span><br><span class="line">fail_device:</span><br><span class="line"> class_destroy(keyirq.class);</span><br><span class="line">fail_class:</span><br><span class="line"> cdev_del(&keyirq.cdev);</span><br><span class="line">fail_cdev:</span><br><span class="line"> unregister_chrdev_region(keyirq.devid, KEYIRQ_CNT);</span><br><span class="line">fail_devid:</span><br><span class="line"> <span class="keyword">return</span> ret;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">static</span> <span class="keyword">void</span> __exit <span class="title">keyirq_exit</span><span class="params">(<span class="keyword">void</span>)</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> i = <span class="number">0</span>;</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">for</span> (i = <span class="number">0</span>; i < KEY_NUM; i++)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 释放中断和IO */</span></span><br><span class="line"> free_irq(keyirq.key[i].irqNum, &keyirq);</span><br><span class="line"> gpio_free(keyirq.key[i].gpio);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 删除定时器 */</span></span><br><span class="line"> del_timer_sync(&keyirq.timer);</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 注销字符设备驱动 */</span></span><br><span class="line"> device_destroy(keyirq.class, keyirq.devid);</span><br><span class="line"> class_destroy(keyirq.class);</span><br><span class="line"> cdev_del(&keyirq.cdev);</span><br><span class="line"> unregister_chrdev_region(keyirq.devid,KEYIRQ_CNT);</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/* 注册驱动和卸载驱动 */</span></span><br><span class="line">module_init(keyirq_init);</span><br><span class="line">module_exit(keyirq_exit);</span><br><span class="line"></span><br><span class="line">MODULE_LICENSE(<span class="string">"GPL"</span>);</span><br><span class="line">MODULE_AUTHOR(<span class="string">"fengyuhang"</span>);</span><br></pre></td></tr></table></figure><ol start="2"><li>测试APP程序</li></ol><figure class="highlight c"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br><span class="line">39</span><br><span class="line">40</span><br><span class="line">41</span><br><span class="line">42</span><br><span class="line">43</span><br><span class="line">44</span><br><span class="line">45</span><br><span class="line">46</span><br><span class="line">47</span><br><span class="line">48</span><br><span class="line">49</span><br><span class="line">50</span><br><span class="line">51</span><br><span class="line">52</span><br><span class="line">53</span><br><span class="line">54</span><br><span class="line">55</span><br><span class="line">56</span><br><span class="line">57</span><br><span class="line">58</span><br><span class="line">59</span><br><span class="line">60</span><br><span class="line">61</span><br><span class="line">62</span><br><span class="line">63</span><br><span class="line">64</span><br><span class="line">65</span><br><span class="line">66</span><br><span class="line">67</span><br><span class="line">68</span><br><span class="line">69</span><br><span class="line">70</span><br><span class="line">71</span><br><span class="line">72</span><br><span class="line">73</span><br><span class="line">74</span><br><span class="line">75</span><br><span class="line">76</span><br><span class="line">77</span><br><span class="line">78</span><br><span class="line">79</span><br><span class="line">80</span><br><span class="line">81</span><br><span class="line">82</span><br><span class="line">83</span><br><span class="line">84</span><br><span class="line">85</span><br><span class="line">86</span><br><span class="line">87</span><br><span class="line">88</span><br><span class="line">89</span><br><span class="line">90</span><br><span class="line">91</span><br><span class="line">92</span><br><span class="line">93</span><br><span class="line">94</span><br><span class="line">95</span><br><span class="line">96</span><br><span class="line">97</span><br><span class="line">98</span><br><span class="line">99</span><br><span class="line">100</span><br><span class="line">101</span><br><span class="line">102</span><br><span class="line">103</span><br><span class="line">104</span><br><span class="line">105</span><br><span class="line">106</span><br><span class="line">107</span><br><span class="line">108</span><br><span class="line">109</span><br><span class="line">110</span><br><span class="line">111</span><br><span class="line">112</span><br><span class="line">113</span><br><span class="line">114</span><br><span class="line">115</span><br><span class="line">116</span><br><span class="line">117</span><br><span class="line">118</span><br><span class="line">119</span><br><span class="line">120</span><br><span class="line">121</span><br></pre></td><td class="code"><pre><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><sys/types.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><sys/stat.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><sys/ioctl.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><sys/select.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><sys/time.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><sys/types.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><poll.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><fcntl.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><unistd.h></span></span></span><br><span class="line"><span class="meta">#<span class="meta-keyword">include</span> <span class="meta-string"><stdio.h></span></span></span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * ./keyReadAPP /dev/keyirq</span></span><br><span class="line"><span class="comment"> * @param argc 应用程序参数个数</span></span><br><span class="line"><span class="comment"> * @param argv 保存的参数,字符串形式。</span></span><br><span class="line"><span class="comment"> * */</span></span><br><span class="line"><span class="function"><span class="keyword">int</span> <span class="title">main</span><span class="params">(<span class="keyword">int</span> argc, <span class="keyword">char</span> *argv[])</span></span></span><br><span class="line"><span class="function"></span>{</span><br><span class="line"> <span class="keyword">int</span> ret = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">int</span> fd = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">char</span> *filename;</span><br><span class="line"> <span class="keyword">unsigned</span> <span class="keyword">char</span> data;</span><br><span class="line"> <span class="comment">/* 使用select */</span></span><br><span class="line"> <span class="comment">// fd_set readfds; /* */</span></span><br><span class="line"> <span class="comment">// struct timeval timeout; /* 超时时间 */</span></span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 使用poll */</span></span><br><span class="line"> <span class="class"><span class="keyword">struct</span> <span class="title">pollfd</span> <span class="title">fds</span>;</span></span><br><span class="line"></span><br><span class="line"><span class="comment">/*</span></span><br><span class="line"><span class="comment"> if (argc != 3)</span></span><br><span class="line"><span class="comment"> {</span></span><br><span class="line"><span class="comment"> printf("输入错误\r\n");</span></span><br><span class="line"><span class="comment"> }</span></span><br><span class="line"><span class="comment">*/</span></span><br><span class="line"></span><br><span class="line"> filename = argv[<span class="number">1</span>];</span><br><span class="line"></span><br><span class="line"> fd = <span class="built_in">open</span>(filename, O_RDWR | O_NONBLOCK); <span class="comment">/* 非阻塞方式打开 */</span></span><br><span class="line"> <span class="keyword">if</span>(fd < <span class="number">0</span>) {</span><br><span class="line"> <span class="built_in">printf</span>(<span class="string">"Error: Open %s fail.\r\n"</span>,filename);</span><br><span class="line"> <span class="keyword">return</span> <span class="number">-1</span>;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 循环读取 */</span></span><br><span class="line"> <span class="keyword">while</span> (<span class="number">1</span>)</span><br><span class="line"> { </span><br><span class="line"> <span class="meta">#<span class="meta-keyword">if</span> 0</span></span><br><span class="line"> <span class="comment">/* 使用select */</span></span><br><span class="line"> FD_ZERO(&readfds);</span><br><span class="line"> FD_SET(fd, &readfds);</span><br><span class="line"> </span><br><span class="line"> timeout.tv_sec = <span class="number">0</span>;</span><br><span class="line"> timeout.tv_usec = <span class="number">500000</span>; <span class="comment">/* 超时时间500毫秒 */</span></span><br><span class="line"> ret = select(fd + <span class="number">1</span>, &readfds, <span class="literal">NULL</span>, <span class="literal">NULL</span>, &timeout);</span><br><span class="line"> <span class="keyword">switch</span> (ret)</span><br><span class="line"> {</span><br><span class="line"> <span class="keyword">case</span> <span class="number">0</span>: <span class="comment">/* 超时 */</span></span><br><span class="line"> <span class="built_in">printf</span>(<span class="string">"select 超时\r\n"</span>);</span><br><span class="line"> <span class="keyword">break</span>;</span><br><span class="line"> <span class="keyword">case</span> <span class="number">-1</span>: <span class="comment">/* 错误 */</span></span><br><span class="line"> <span class="keyword">break</span>;</span><br><span class="line"> <span class="keyword">default</span>: <span class="comment">/* 可以读取数据 */</span></span><br><span class="line"> <span class="keyword">if</span>(FD_ISSET(fd, &readfds))</span><br><span class="line"> {</span><br><span class="line"> ret = <span class="built_in">read</span>(fd, &data, <span class="keyword">sizeof</span>(data));</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> { </span><br><span class="line"> </span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> <span class="built_in">printf</span>(<span class="string">"keyValue = %#x\r\n"</span>, data);</span><br><span class="line"> }</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">break</span>;</span><br><span class="line"> }</span><br><span class="line"> <span class="meta">#<span class="meta-keyword">endif</span></span></span><br><span class="line"></span><br><span class="line"> <span class="comment">/* 使用poll */</span></span><br><span class="line"> fds.fd = fd;</span><br><span class="line"> fds.events = POLLIN;</span><br><span class="line"> ret = poll(&fds, <span class="number">1</span>, <span class="number">500</span>); <span class="comment">/* 超时时间500ms */</span></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">if</span> (ret == <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 超时 */</span></span><br><span class="line"> <span class="built_in">printf</span>(<span class="string">"poll 超时\r\n"</span>);</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 错误 */</span></span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 可以读取 */</span></span><br><span class="line"> <span class="keyword">if</span> (fds.revents | POLLIN)</span><br><span class="line"> {</span><br><span class="line"> <span class="comment">/* 可读取 */</span></span><br><span class="line"> ret = <span class="built_in">read</span>(fd, &data, <span class="keyword">sizeof</span>(data));</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> { </span><br><span class="line"> </span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">else</span></span><br><span class="line"> {</span><br><span class="line"> <span class="built_in">printf</span>(<span class="string">"keyValue = %#x\r\n"</span>, data);</span><br><span class="line"> }</span><br><span class="line"> }</span><br><span class="line"> }</span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> <span class="comment">/* 关闭 */</span></span><br><span class="line"> ret = <span class="built_in">close</span>(fd);</span><br><span class="line"> <span class="keyword">if</span> (ret < <span class="number">0</span>)</span><br><span class="line"> {</span><br><span class="line"> <span class="built_in">printf</span>(<span class="string">"Error: Close %s fail.\r\n"</span>, filename);</span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line">}</span><br></pre></td></tr></table></figure><div class="declare"><ul class="post-copyright"><li><strong>本文作者:</strong> 路痴的兔子</li><li><strong>本文链接:</strong> <a href="https:/proudrabbit.gitee.io/IMX6ULL嵌入式Linux驱动学习笔记(十).html" title="IMX6ULL嵌入式Linux驱动学习笔记(十)" target="_blank">https:/proudrabbit.gitee.io/IMX6ULL嵌入式Linux驱动学习笔记(十).html</a></li><li><strong>版权声明: </strong>本博客所有文章除特别声明外,均采用 <a href="/~https://github.com/JoeyBling/hexo-theme-yilia-plus/blob/master/LICENSE" rel="external nofollow" target="_blank">MIT</a> 许可协议。转载请注明出处!</li></ul></div></div><div class="article-info article-info-index"><div class="article-tag tagcloud"><i class="icon-price-tags icon"></i><ul class="article-tag-list"><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag article-tag-list-link color4">嵌入式</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag article-tag-list-link color3">IMX6ULL</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag article-tag-list-link color5">学习笔记</a></li></ul></div><div class="share-btn share-icons tooltip-left"><div class="tooltip tooltip-east"><span class="tooltip-item"><a href="javascript:;" class="share-sns share-outer"><i class="icon icon-share"></i> </a></span><span class="tooltip-content"><div class="share-wrap"><div class="share-icons"><a class="weibo share-sns" href="javascript:;" data-type="weibo"><i class="icon icon-weibo"></i> </a><a class="weixin share-sns wxFab" href="javascript:;" data-type="weixin"><i class="icon icon-weixin"></i> </a><a class="qq share-sns" href="javascript:;" data-type="qq"><i class="icon icon-qq"></i> </a><a class="douban share-sns" href="javascript:;" data-type="douban"><i class="icon icon-douban"></i> </a><a class="qzone share-sns" href="javascript:;" data-type="qzone"><i class="icon icon-qzone"></i> </a><a class="facebook share-sns" href="javascript:;" data-type="facebook"><i class="icon icon-facebook"></i> </a><a class="twitter share-sns" href="javascript:;" data-type="twitter"><i class="icon icon-twitter"></i> </a><a class="google share-sns" href="javascript:;" data-type="google"><i class="icon icon-google"></i></a></div></div></span></div></div><div class="page-modal wx-share js-wx-box"><a class="close js-modal-close" href="javascript:;"><i class="icon icon-close"></i></a><p>扫一扫,分享到微信</p><div class="wx-qrcode"><img src="//api.qrserver.com/v1/create-qr-code/?size=150x150&data=https:/ProudRabbit.gitee.io/gitee.io/IMX6ULL%E5%B5%8C%E5%85%A5%E5%BC%8FLinux%E9%A9%B1%E5%8A%A8%E5%AD%A6%E4%B9%A0%E7%AC%94%E8%AE%B0%EF%BC%88%E5%8D%81%EF%BC%89.html" alt="微信分享二维码"></div></div><div class="mask js-mask"></div><div class="clearfix"></div></div></div></article><nav id="article-nav"><a href="/IMX6ULL%E5%B5%8C%E5%85%A5%E5%BC%8FLinux%E9%A9%B1%E5%8A%A8%E5%AD%A6%E4%B9%A0%E7%AC%94%E8%AE%B0%EF%BC%88%E5%8D%81%E4%B8%80%EF%BC%89.html" id="article-nav-newer" class="article-nav-link-wrap"><i class="icon-circle-left"></i><div class="article-nav-title">IMX6ULL嵌入式Linux驱动学习笔记(十一)</div></a><a href="/IMX6ULL%E5%B5%8C%E5%85%A5%E5%BC%8FLinux%E9%A9%B1%E5%8A%A8%E5%AD%A6%E4%B9%A0%E7%AC%94%E8%AE%B0%EF%BC%88%E4%B9%9D%EF%BC%89.html" id="article-nav-older" class="article-nav-link-wrap"><div class="article-nav-title">IMX6ULL嵌入式Linux驱动学习笔记(九)</div><i class="icon-circle-right"></i></a></nav><aside class="wrap-side-operation"><div class="mod-side-operation"><div class="jump-container" id="js-jump-container" style="display:none"><a href="javascript:void(0)" class="mod-side-operation__jump-to-top"><i class="icon-font icon-back"></i></a><div id="js-jump-plan-container" class="jump-plan-container" style="top:-11px"><i class="icon-font icon-plane jump-plane"></i></div></div><div class="toc-container tooltip-left"><i class="icon-font icon-category"></i><div class="tooltip tooltip-east"><span class="tooltip-item"></span> <span class="tooltip-content"><div class="toc-article"><ol class="toc"><li class="toc-item toc-level-2"><a class="toc-link" href="#一、Linux阻塞和非阻塞IO"><span class="toc-number">1.</span> <span class="toc-text">一、Linux阻塞和非阻塞IO</span></a><ol class="toc-child"><li class="toc-item toc-level-3"><a class="toc-link" href="#1-1-阻塞和非阻塞简介"><span class="toc-number">1.1.</span> <span class="toc-text">1.1 阻塞和非阻塞简介</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#1-2-等待队列(阻塞访问)"><span class="toc-number">1.2.</span> <span class="toc-text">1.2 等待队列(阻塞访问)</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#1-3-轮询(非阻塞访问)"><span class="toc-number">1.3.</span> <span class="toc-text">1.3 轮询(非阻塞访问)</span></a></li></ol></li><li class="toc-item toc-level-2"><a class="toc-link" href="#二、编写试验驱动"><span class="toc-number">2.</span> <span class="toc-text">二、编写试验驱动</span></a><ol class="toc-child"><li class="toc-item toc-level-3"><a class="toc-link" href="#2-1-阻塞式访问驱动"><span class="toc-number">2.1.</span> <span class="toc-text">2.1 阻塞式访问驱动</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#2-2-非阻塞式访问"><span class="toc-number">2.2.</span> <span class="toc-text">2.2 非阻塞式访问</span></a></li></ol></li></ol></div></span></div></div></div></aside><div id="beaudar-comment"><link rel="stylesheet" href="/lib/beaudar.css"><script src="https://beaudar.lipk.org/client.js" repo="ProudRabbit/ProudRabbit.github.io" issue-term="title" label="comment" theme="github-light" crossorigin="anonymous" async></script></div></div></div></div><footer id="footer"><div class="outer"><div id="footer-info"><div class="footer-left">© 2020-2023 <a href="https:/proudrabbit.gitee.io/" target="_blank">路痴的兔子</a></div><div class="footer-right">由<a href="https://hexo.io/" target="_blank">Hexo</a>强力驱动 <span>|</span> 主题:<a href="/~https://github.com/JoeyBling/hexo-theme-yilia-plus" target="_blank">yilia-plus</a> <span>|</span> 评论插件:<a href="https://beaudar.lipk.org" target="_blank">Beaudar(表达)</a></div></div></div><script src="/lib/busuanzi.pure.js"></script><span id="busuanzi_container_site_pv" style="display:none">本站总访问量<span id="busuanzi_value_site_pv"></span>次 <span class="post-meta-divider">|</span> </span><span id="busuanzi_container_site_uv" style="display:none">本站访客数<span id="busuanzi_value_site_uv"></span>人</span></footer></div><script>var yiliaConfig={mathjax:!1,isHome:!1,isPost:!0,isArchive:!1,isTag:!1,isCategory:!1,open_in_new:!0,toc_hide_index:!0,root:"/",innerArchive:!0,showTags:!0}</script><script>!function(r){function e(t){if(i[t])return i[t].exports;var n=i[t]={exports:{},id:t,loaded:!1};return r[t].call(n.exports,n,n.exports,e),n.loaded=!0,n.exports}var i={};e.m=r,e.c=i,e.p="./",e(0)}([function(t,n,r){r(208),t.exports=r(205)},function(t,n,r){var d=r(3),y=r(46),g=r(26),b=r(27),x=r(47),m="prototype",S=function(t,n,r){var e,i,o,u,c=t&S.F,f=t&S.G,a=t&S.S,s=t&S.P,l=t&S.B,h=f?d:a?d[n]||(d[n]={}):(d[n]||{})[m],v=f?y:y[n]||(y[n]={}),p=v[m]||(v[m]={});for(e in f&&(r=n),r)o=((i=!c&&h&&void 0!==h[e])?h:r)[e],u=l&&i?x(o,d):s&&"function"==typeof o?x(Function.call,o):o,h&&b(h,e,o,t&S.U),v[e]!=o&&g(v,e,u),s&&p[e]!=o&&(p[e]=o)};d.core=y,S.F=1,S.G=2,S.S=4,S.P=8,S.B=16,S.W=32,S.U=64,S.R=128,t.exports=S},function(t,n,r){var e=r(5);t.exports=function(t){if(!e(t))throw TypeError(t+" is not an object!");return t}},function(t,n){var r=t.exports="undefined"!=typeof window&&window.Math==Math?window:"undefined"!=typeof self&&self.Math==Math?self:Function("return this")();"number"==typeof __g&&(__g=r)},function(t,n){t.exports=function(t){try{return!!t()}catch(t){return!0}}},function(t,n){t.exports=function(t){return"object"==typeof t?null!==t:"function"==typeof t}},function(t,n){var r=t.exports="undefined"!=typeof window&&window.Math==Math?window:"undefined"!=typeof self&&self.Math==Math?self:Function("return this")();"number"==typeof __g&&(__g=r)},function(t,n,r){var e=r(118)("wks"),i=r(79),o=r(3).Symbol,u="function"==typeof o;(t.exports=function(t){return e[t]||(e[t]=u&&o[t]||(u?o:i)("Symbol."+t))}).store=e},function(t,n,r){var e=r(49),i=Math.min;t.exports=function(t){return 0<t?i(e(t),9007199254740991):0}},function(t,n){var r={}.hasOwnProperty;t.exports=function(t,n){return r.call(t,n)}},function(t,n,r){t.exports=!r(4)(function(){return 7!=Object.defineProperty({},"a",{get:function(){return 7}}).a})},function(t,n,r){var e=r(2),i=r(174),o=r(53),u=Object.defineProperty;n.f=r(10)?Object.defineProperty:function(t,n,r){if(e(t),n=o(n,!0),e(r),i)try{return u(t,n,r)}catch(t){}if("get"in r||"set"in r)throw TypeError("Accessors not supported!");return"value"in r&&(t[n]=r.value),t}},function(t,n,r){t.exports=!r(20)(function(){return 7!=Object.defineProperty({},"a",{get:function(){return 7}}).a})},function(t,n,r){var e=r(14),i=r(24);t.exports=r(12)?function(t,n,r){return e.f(t,n,i(1,r))}:function(t,n,r){return t[n]=r,t}},function(t,n,r){var e=r(22),i=r(60),o=r(42),u=Object.defineProperty;n.f=r(12)?Object.defineProperty:function(t,n,r){if(e(t),n=o(n,!0),e(r),i)try{return u(t,n,r)}catch(t){}if("get"in r||"set"in r)throw TypeError("Accessors not supported!");return"value"in r&&(t[n]=r.value),t}},function(t,n,r){var e=r(96),i=r(34);t.exports=function(t){return e(i(t))}},function(t,n,r){var e=r(40)("wks"),i=r(25),o=r(6).Symbol,u="function"==typeof o;(t.exports=function(t){return e[t]||(e[t]=u&&o[t]||(u?o:i)("Symbol."+t))}).store=e},function(t,n,r){var e=r(51);t.exports=function(t){return Object(e(t))}},function(t,n){t.exports=function(t){return"object"==typeof t?null!==t:"function"==typeof t}},function(t,n){var r=t.exports={version:"2.6.9"};"number"==typeof __e&&(__e=r)},function(t,n){t.exports=function(t){try{return!!t()}catch(t){return!0}}},function(t,n){t.exports=function(t){if("function"!=typeof t)throw TypeError(t+" is not a function!");return t}},function(t,n,r){var e=r(18);t.exports=function(t){if(!e(t))throw TypeError(t+" is not an object!");return t}},function(t,n){t.exports=!0},function(t,n){t.exports=function(t,n){return{enumerable:!(1&t),configurable:!(2&t),writable:!(4&t),value:n}}},function(t,n){var r=0,e=Math.random();t.exports=function(t){return"Symbol(".concat(void 0===t?"":t,")_",(++r+e).toString(36))}},function(t,n,r){var e=r(11),i=r(75);t.exports=r(10)?function(t,n,r){return e.f(t,n,i(1,r))}:function(t,n,r){return t[n]=r,t}},function(t,n,r){var o=r(3),u=r(26),c=r(30),f=r(79)("src"),e=r(219),i="toString",a=(""+e).split(i);r(46).inspectSource=function(t){return e.call(t)},(t.exports=function(t,n,r,e){var i="function"==typeof r;i&&(c(r,"name")||u(r,"name",n)),t[n]!==r&&(i&&(c(r,f)||u(r,f,t[n]?""+t[n]:a.join(String(n)))),t===o?t[n]=r:e?t[n]?t[n]=r:u(t,n,r):(delete t[n],u(t,n,r)))})(Function.prototype,i,function(){return"function"==typeof this&&this[f]||e.call(this)})},function(t,n,r){function e(t,n,r,e){var i=String(u(t)),o="<"+n;return""!==r&&(o+=" "+r+'="'+String(e).replace(c,""")+'"'),o+">"+i+"</"+n+">"}var i=r(1),o=r(4),u=r(51),c=/"/g;t.exports=function(n,t){var r={};r[n]=t(e),i(i.P+i.F*o(function(){var t=""[n]('"');return t!==t.toLowerCase()||3<t.split('"').length}),"String",r)}},function(t,n,r){var e=r(65),i=r(35);t.exports=Object.keys||function(t){return e(t,i)}},function(t,n){var r={}.hasOwnProperty;t.exports=function(t,n){return r.call(t,n)}},function(t,n,r){var e=r(117),i=r(75),o=r(33),u=r(53),c=r(30),f=r(174),a=Object.getOwnPropertyDescriptor;n.f=r(10)?a:function(t,n){if(t=o(t),n=u(n,!0),f)try{return a(t,n)}catch(t){}if(c(t,n))return i(!e.f.call(t,n),t[n])}},function(t,n,r){var e=r(30),i=r(17),o=r(154)("IE_PROTO"),u=Object.prototype;t.exports=Object.getPrototypeOf||function(t){return t=i(t),e(t,o)?t[o]:"function"==typeof t.constructor&&t instanceof t.constructor?t.constructor.prototype:t instanceof Object?u:null}},function(t,n,r){var e=r(116),i=r(51);t.exports=function(t){return e(i(t))}},function(t,n){t.exports=function(t){if(null==t)throw TypeError("Can't call method on "+t);return t}},function(t,n){t.exports="constructor,hasOwnProperty,isPrototypeOf,propertyIsEnumerable,toLocaleString,toString,valueOf".split(",")},function(t,n){t.exports={}},function(t,n){n.f={}.propertyIsEnumerable},function(t,n,r){var e=r(14).f,i=r(9),o=r(16)("toStringTag");t.exports=function(t,n,r){t&&!i(t=r?t:t.prototype,o)&&e(t,o,{configurable:!0,value:n})}},function(t,n,r){var e=r(40)("keys"),i=r(25);t.exports=function(t){return e[t]||(e[t]=i(t))}},function(t,n,r){var e=r(19),i=r(6),o="__core-js_shared__",u=i[o]||(i[o]={});(t.exports=function(t,n){return u[t]||(u[t]=void 0!==n?n:{})})("versions",[]).push({version:e.version,mode:r(23)?"pure":"global",copyright:"© 2019 Denis Pushkarev (zloirock.ru)"})},function(t,n){var r=Math.ceil,e=Math.floor;t.exports=function(t){return isNaN(t=+t)?0:(0<t?e:r)(t)}},function(t,n,r){var i=r(18);t.exports=function(t,n){if(!i(t))return t;var r,e;if(n&&"function"==typeof(r=t.toString)&&!i(e=r.call(t)))return e;if("function"==typeof(r=t.valueOf)&&!i(e=r.call(t)))return e;if(!n&&"function"==typeof(r=t.toString)&&!i(e=r.call(t)))return e;throw TypeError("Can't convert object to primitive value")}},function(t,n,r){var e=r(6),i=r(19),o=r(23),u=r(44),c=r(14).f;t.exports=function(t){var n=i.Symbol||(i.Symbol=!o&&e.Symbol||{});"_"==t.charAt(0)||t in n||c(n,t,{value:u.f(t)})}},function(t,n,r){n.f=r(16)},function(t,n){var r={}.toString;t.exports=function(t){return r.call(t).slice(8,-1)}},function(t,n){var r=t.exports={version:"2.6.9"};"number"==typeof __e&&(__e=r)},function(t,n,r){var o=r(21);t.exports=function(e,i,t){if(o(e),void 0===i)return e;switch(t){case 1:return function(t){return e.call(i,t)};case 2:return function(t,n){return e.call(i,t,n)};case 3:return function(t,n,r){return e.call(i,t,n,r)}}return function(){return e.apply(i,arguments)}}},function(t,n,r){"use strict";var e=r(4);t.exports=function(t,n){return!!t&&e(function(){n?t.call(null,function(){},1):t.call(null)})}},function(t,n){var r=Math.ceil,e=Math.floor;t.exports=function(t){return isNaN(t=+t)?0:(0<t?e:r)(t)}},function(t,n,r){var x=r(47),m=r(116),S=r(17),w=r(8),e=r(138);t.exports=function(l,t){var h=1==l,v=2==l,p=3==l,d=4==l,y=6==l,g=5==l||y,b=t||e;return function(t,n,r){for(var e,i,o=S(t),u=m(o),c=x(n,r,3),f=w(u.length),a=0,s=h?b(t,f):v?b(t,0):void 0;a<f;a++)if((g||a in u)&&(i=c(e=u[a],a,o),l))if(h)s[a]=i;else if(i)switch(l){case 3:return!0;case 5:return e;case 6:return a;case 2:s.push(e)}else if(d)return!1;return y?-1:p||d?d:s}}},function(t,n){t.exports=function(t){if(null==t)throw TypeError("Can't call method on "+t);return t}},function(t,n,r){var i=r(1),o=r(46),u=r(4);t.exports=function(t,n){var r=(o.Object||{})[t]||Object[t],e={};e[t]=n(r),i(i.S+i.F*u(function(){r(1)}),"Object",e)}},function(t,n,r){var i=r(5);t.exports=function(t,n){if(!i(t))return t;var r,e;if(n&&"function"==typeof(r=t.toString)&&!i(e=r.call(t)))return e;if("function"==typeof(r=t.valueOf)&&!i(e=r.call(t)))return e;if(!n&&"function"==typeof(r=t.toString)&&!i(e=r.call(t)))return e;throw TypeError("Can't convert object to primitive value")}},function(t,n,r){var d=r(6),y=r(19),g=r(93),b=r(13),x=r(9),m="prototype",S=function(t,n,r){var e,i,o,u=t&S.F,c=t&S.G,f=t&S.S,a=t&S.P,s=t&S.B,l=t&S.W,h=c?y:y[n]||(y[n]={}),v=h[m],p=c?d:f?d[n]:(d[n]||{})[m];for(e in c&&(r=n),r)(i=!u&&p&&void 0!==p[e])&&x(h,e)||(o=i?p[e]:r[e],h[e]=c&&"function"!=typeof p[e]?r[e]:s&&i?g(o,d):l&&p[e]==o?function(e){function t(t,n,r){if(this instanceof e){switch(arguments.length){case 0:return new e;case 1:return new e(t);case 2:return new e(t,n)}return new e(t,n,r)}return e.apply(this,arguments)}return t[m]=e[m],t}(o):a&&"function"==typeof o?g(Function.call,o):o,a&&((h.virtual||(h.virtual={}))[e]=o,t&S.R&&v&&!v[e]&&b(v,e,o)))};S.F=1,S.G=2,S.S=4,S.P=8,S.B=16,S.W=32,S.U=64,S.R=128,t.exports=S},function(t,n,r){var e=r(34);t.exports=function(t){return Object(e(t))}},function(t,n,r){function i(t,n,r){var e=c.get(t);if(!e){if(!r)return;c.set(t,e=new o)}var i=e.get(n);if(!i){if(!r)return;e.set(n,i=new o)}return i}var o=r(196),e=r(1),u=r(118)("metadata"),c=u.store||(u.store=new(r(200)));t.exports={store:c,map:i,has:function(t,n,r){var e=i(n,r,!1);return void 0!==e&&e.has(t)},get:function(t,n,r){var e=i(n,r,!1);return void 0===e?void 0:e.get(t)},set:function(t,n,r,e){i(r,e,!0).set(t,n)},keys:function(t,n){var r=i(t,n,!1),e=[];return r&&r.forEach(function(t,n){e.push(n)}),e},key:function(t){return void 0===t||"symbol"==typeof t?t:String(t)},exp:function(t){e(e.S,"Reflect",t)}}},function(t,n,r){"use strict";var s,g,b,x,m,e,h,S,i,w,o,u,_,O,c,f,a,E,M,v,p,P,j,F,d,l,y,A,I,L,N,T,k,R,C,D,G,W,U,V,B,q,z,K,H,J,$,Y,X,Q,Z,tt,nt,rt,et,it,ot,ut,ct,ft,at,st,lt,ht,vt,pt,dt,yt,gt,bt,xt,mt,St,wt,_t,Ot,Et,Mt,Pt,jt,Ft,At,It,Lt,Nt,Tt,kt,Rt,Ct,Dt,Gt,Wt,Ut,Vt,Bt,qt,zt,Kt;r(10)?(s=r(68),g=r(3),b=r(4),x=r(1),m=r(132),e=r(159),h=r(47),S=r(70),i=r(75),w=r(26),o=r(76),u=r(49),_=r(8),O=r(194),c=r(78),f=r(53),a=r(30),E=r(81),M=r(5),v=r(17),p=r(145),P=r(72),j=r(32),F=r(73).f,d=r(161),l=r(79),y=r(7),A=r(50),I=r(120),L=r(119),N=r(162),T=r(82),k=r(125),R=r(77),C=r(137),D=r(166),G=r(11),W=r(31),U=G.f,V=W.f,B=g.RangeError,q=g.TypeError,z=g.Uint8Array,H="Shared"+(K="ArrayBuffer"),J="BYTES_PER_ELEMENT",$="prototype",Y=Array[$],X=e.ArrayBuffer,Q=e.DataView,Z=A(0),tt=A(2),nt=A(3),rt=A(4),et=A(5),it=A(6),ot=I(!0),ut=I(!1),ct=N.values,ft=N.keys,at=N.entries,st=Y.lastIndexOf,lt=Y.reduce,ht=Y.reduceRight,vt=Y.join,pt=Y.sort,dt=Y.slice,yt=Y.toString,gt=Y.toLocaleString,bt=y("iterator"),xt=y("toStringTag"),mt=l("typed_constructor"),St=l("def_constructor"),wt=m.CONSTR,_t=m.TYPED,Ot=m.VIEW,Et="Wrong length!",Mt=A(1,function(t,n){return It(L(t,t[St]),n)}),Pt=b(function(){return 1===new z(new Uint16Array([1]).buffer)[0]}),jt=!!z&&!!z[$].set&&b(function(){new z(1).set({})}),Ft=function(t,n){var r=u(t);if(r<0||r%n)throw B("Wrong offset!");return r},At=function(t){if(M(t)&&_t in t)return t;throw q(t+" is not a typed array!")},It=function(t,n){if(!(M(t)&&mt in t))throw q("It is not a typed array constructor!");return new t(n)},Lt=function(t,n){return Nt(L(t,t[St]),n)},Nt=function(t,n){for(var r=0,e=n.length,i=It(t,e);r<e;)i[r]=n[r++];return i},Tt=function(t,n,r){U(t,n,{get:function(){return this._d[r]}})},kt=function(t){var n,r,e,i,o,u,c=v(t),f=arguments.length,a=1<f?arguments[1]:void 0,s=void 0!==a,l=d(c);if(null!=l&&!p(l)){for(u=l.call(c),e=[],n=0;!(o=u.next()).done;n++)e.push(o.value);c=e}for(s&&2<f&&(a=h(a,arguments[2],2)),n=0,r=_(c.length),i=It(this,r);n<r;n++)i[n]=s?a(c[n],n):c[n];return i},Rt=function(){for(var t=0,n=arguments.length,r=It(this,n);t<n;)r[t]=arguments[t++];return r},Ct=!!z&&b(function(){gt.call(new z(1))}),Dt=function(){return gt.apply(Ct?dt.call(At(this)):At(this),arguments)},Gt={copyWithin:function(t,n){return D.call(At(this),t,n,2<arguments.length?arguments[2]:void 0)},every:function(t){return rt(At(this),t,1<arguments.length?arguments[1]:void 0)},fill:function(t){return C.apply(At(this),arguments)},filter:function(t){return Lt(this,tt(At(this),t,1<arguments.length?arguments[1]:void 0))},find:function(t){return et(At(this),t,1<arguments.length?arguments[1]:void 0)},findIndex:function(t){return it(At(this),t,1<arguments.length?arguments[1]:void 0)},forEach:function(t){Z(At(this),t,1<arguments.length?arguments[1]:void 0)},indexOf:function(t){return ut(At(this),t,1<arguments.length?arguments[1]:void 0)},includes:function(t){return ot(At(this),t,1<arguments.length?arguments[1]:void 0)},join:function(t){return vt.apply(At(this),arguments)},lastIndexOf:function(t){return st.apply(At(this),arguments)},map:function(t){return Mt(At(this),t,1<arguments.length?arguments[1]:void 0)},reduce:function(t){return lt.apply(At(this),arguments)},reduceRight:function(t){return ht.apply(At(this),arguments)},reverse:function(){for(var t,n=this,r=At(n).length,e=Math.floor(r/2),i=0;i<e;)t=n[i],n[i++]=n[--r],n[r]=t;return n},some:function(t){return nt(At(this),t,1<arguments.length?arguments[1]:void 0)},sort:function(t){return pt.call(At(this),t)},subarray:function(t,n){var r=At(this),e=r.length,i=c(t,e);return new(L(r,r[St]))(r.buffer,r.byteOffset+i*r.BYTES_PER_ELEMENT,_((void 0===n?e:c(n,e))-i))}},Wt=function(t,n){return Lt(this,dt.call(At(this),t,n))},Ut=function(t){At(this);var n=Ft(arguments[1],1),r=this.length,e=v(t),i=_(e.length),o=0;if(r<i+n)throw B(Et);for(;o<i;)this[n+o]=e[o++]},Vt={entries:function(){return at.call(At(this))},keys:function(){return ft.call(At(this))},values:function(){return ct.call(At(this))}},Bt=function(t,n){return M(t)&&t[_t]&&"symbol"!=typeof n&&n in t&&String(+n)==String(n)},qt=function(t,n){return Bt(t,n=f(n,!0))?i(2,t[n]):V(t,n)},zt=function(t,n,r){return!(Bt(t,n=f(n,!0))&&M(r)&&a(r,"value"))||a(r,"get")||a(r,"set")||r.configurable||a(r,"writable")&&!r.writable||a(r,"enumerable")&&!r.enumerable?U(t,n,r):(t[n]=r.value,t)},wt||(W.f=qt,G.f=zt),x(x.S+x.F*!wt,"Object",{getOwnPropertyDescriptor:qt,defineProperty:zt}),b(function(){yt.call({})})&&(yt=gt=function(){return vt.call(this)}),Kt=o({},Gt),o(Kt,Vt),w(Kt,bt,Vt.values),o(Kt,{slice:Wt,set:Ut,constructor:function(){},toString:yt,toLocaleString:Dt}),Tt(Kt,"buffer","b"),Tt(Kt,"byteOffset","o"),Tt(Kt,"byteLength","l"),Tt(Kt,"length","e"),U(Kt,xt,{get:function(){return this[_t]}}),t.exports=function(t,l,n,h){var v=t+((h=!!h)?"Clamped":"")+"Array",p="get"+t,d="set"+t,y=g[v],o=y||{},r=y&&j(y),e=!y||!m.ABV,i={},u=y&&y[$];e?(y=n(function(t,n,r,e){S(t,y,v,"_d");var i,o,u,c,f=0,a=0;if(M(n)){if(!(n instanceof X||(c=E(n))==K||c==H))return _t in n?Nt(y,n):kt.call(y,n);i=n,a=Ft(r,l);var s=n.byteLength;if(void 0===e){if(s%l)throw B(Et);if((o=s-a)<0)throw B(Et)}else if(s<(o=_(e)*l)+a)throw B(Et);u=o/l}else u=O(n),i=new X(o=u*l);for(w(t,"_d",{b:i,o:a,l:o,e:u,v:new Q(i)});f<u;)!function(i){U(t,i,{get:function(){return(t=this._d).v[p](i*l+t.o,Pt);var t},set:function(t){return n=i,r=t,e=this._d,h&&(r=(r=Math.round(r))<0?0:255<r?255:255&r),void e.v[d](n*l+e.o,r,Pt);var n,r,e},enumerable:!0})}(f++)}),u=y[$]=P(Kt),w(u,"constructor",y)):b(function(){y(1)})&&b(function(){new y(-1)})&&k(function(t){new y,new y(null),new y(1.5),new y(t)},!0)||(y=n(function(t,n,r,e){var i;return S(t,y,v),M(n)?n instanceof X||(i=E(n))==K||i==H?void 0!==e?new o(n,Ft(r,l),e):void 0!==r?new o(n,Ft(r,l)):new o(n):_t in n?Nt(y,n):kt.call(y,n):new o(O(n))}),Z(r!==Function.prototype?F(o).concat(F(r)):F(o),function(t){t in y||w(y,t,o[t])}),y[$]=u,s||(u.constructor=y));var c=u[bt],f=!!c&&("values"==c.name||null==c.name),a=Vt.values;w(y,mt,!0),w(u,_t,v),w(u,Ot,!0),w(u,St,y),(h?new y(1)[xt]==v:xt in u)||U(u,xt,{get:function(){return v}}),i[v]=y,x(x.G+x.W+x.F*(y!=o),i),x(x.S,v,{BYTES_PER_ELEMENT:l}),x(x.S+x.F*b(function(){o.of.call(y,1)}),v,{from:kt,of:Rt}),J in u||w(u,J,l),x(x.P,v,Gt),R(v),x(x.P+x.F*jt,v,{set:Ut}),x(x.P+x.F*!f,v,Vt),s||u.toString==yt||(u.toString=yt),x(x.P+x.F*b(function(){new y(1).slice()}),v,{slice:Wt}),x(x.P+x.F*(b(function(){return[1,2].toLocaleString()!=new y([1,2]).toLocaleString()})||!b(function(){u.toLocaleString.call([1,2])})),v,{toLocaleString:Dt}),T[v]=f?c:a,s||f||w(u,bt,a)}):t.exports=function(){}},function(t,n){var r={}.toString;t.exports=function(t){return r.call(t).slice(8,-1)}},function(t,n,r){var e=r(18),i=r(6).document,o=e(i)&&e(i.createElement);t.exports=function(t){return o?i.createElement(t):{}}},function(t,n,r){t.exports=!r(12)&&!r(20)(function(){return 7!=Object.defineProperty(r(59)("div"),"a",{get:function(){return 7}}).a})},function(t,n,r){"use strict";function x(){return this}var m=r(23),S=r(54),w=r(66),_=r(13),O=r(36),E=r(98),M=r(38),P=r(104),j=r(16)("iterator"),F=!([].keys&&"next"in[].keys()),A="values";t.exports=function(t,n,r,e,i,o,u){function c(t){if(!F&&t in p)return p[t];switch(t){case"keys":case A:return function(){return new r(this,t)}}return function(){return new r(this,t)}}E(r,n,e);var f,a,s,l=n+" Iterator",h=i==A,v=!1,p=t.prototype,d=p[j]||p["@@iterator"]||i&&p[i],y=d||c(i),g=i?h?c("entries"):y:void 0,b="Array"==n&&p.entries||d;if(b&&(s=P(b.call(new t)))!==Object.prototype&&s.next&&(M(s,l,!0),m||"function"==typeof s[j]||_(s,j,x)),h&&d&&d.name!==A&&(v=!0,y=function(){return d.call(this)}),m&&!u||!F&&!v&&p[j]||_(p,j,y),O[n]=y,O[l]=x,i)if(f={values:h?y:c(A),keys:o?y:c("keys"),entries:g},u)for(a in f)a in p||w(p,a,f[a]);else S(S.P+S.F*(F||v),n,f);return f}},function(t,n,e){function i(){}var o=e(22),u=e(101),c=e(35),f=e(39)("IE_PROTO"),a="prototype",s=function(){var t,n=e(59)("iframe"),r=c.length;for(n.style.display="none",e(95).appendChild(n),n.src="javascript:",(t=n.contentWindow.document).open(),t.write("<script>document.F=Object<\/script>"),t.close(),s=t.F;r--;)delete s[a][c[r]];return s()};t.exports=Object.create||function(t,n){var r;return null!==t?(i[a]=o(t),r=new i,i[a]=null,r[f]=t):r=s(),void 0===n?r:u(r,n)}},function(t,n,r){var e=r(65),i=r(35).concat("length","prototype");n.f=Object.getOwnPropertyNames||function(t){return e(t,i)}},function(t,n){n.f=Object.getOwnPropertySymbols},function(t,n,r){var u=r(9),c=r(15),f=r(92)(!1),a=r(39)("IE_PROTO");t.exports=function(t,n){var r,e=c(t),i=0,o=[];for(r in e)r!=a&&u(e,r)&&o.push(r);for(;n.length>i;)u(e,r=n[i++])&&(~f(o,r)||o.push(r));return o}},function(t,n,r){t.exports=r(13)},function(t,n,r){var e=r(7)("unscopables"),i=Array.prototype;null==i[e]&&r(26)(i,e,{}),t.exports=function(t){i[e][t]=!0}},function(t,n){t.exports=!1},function(t,n,r){function e(t){c(t,i,{value:{i:"O"+ ++f,w:{}}})}var i=r(79)("meta"),o=r(5),u=r(30),c=r(11).f,f=0,a=Object.isExtensible||function(){return!0},s=!r(4)(function(){return a(Object.preventExtensions({}))}),l=t.exports={KEY:i,NEED:!1,fastKey:function(t,n){if(!o(t))return"symbol"==typeof t?t:("string"==typeof t?"S":"P")+t;if(!u(t,i)){if(!a(t))return"F";if(!n)return"E";e(t)}return t[i].i},getWeak:function(t,n){if(!u(t,i)){if(!a(t))return!0;if(!n)return!1;e(t)}return t[i].w},onFreeze:function(t){return s&&l.NEED&&a(t)&&!u(t,i)&&e(t),t}}},function(t,n){t.exports=function(t,n,r,e){if(!(t instanceof n)||void 0!==e&&e in t)throw TypeError(r+": incorrect invocation!");return t}},function(t,n,r){var h=r(47),v=r(177),p=r(145),d=r(2),y=r(8),g=r(161),b={},x={};(n=t.exports=function(t,n,r,e,i){var o,u,c,f,a=i?function(){return t}:g(t),s=h(r,e,n?2:1),l=0;if("function"!=typeof a)throw TypeError(t+" is not iterable!");if(p(a)){for(o=y(t.length);l<o;l++)if((f=n?s(d(u=t[l])[0],u[1]):s(t[l]))===b||f===x)return f}else for(c=a.call(t);!(u=c.next()).done;)if((f=v(c,s,u.value,n))===b||f===x)return f}).BREAK=b,n.RETURN=x},function(t,n,e){function i(){}var o=e(2),u=e(183),c=e(141),f=e(154)("IE_PROTO"),a="prototype",s=function(){var t,n=e(140)("iframe"),r=c.length;for(n.style.display="none",e(143).appendChild(n),n.src="javascript:",(t=n.contentWindow.document).open(),t.write("<script>document.F=Object<\/script>"),t.close(),s=t.F;r--;)delete s[a][c[r]];return s()};t.exports=Object.create||function(t,n){var r;return null!==t?(i[a]=o(t),r=new i,i[a]=null,r[f]=t):r=s(),void 0===n?r:u(r,n)}},function(t,n,r){var e=r(185),i=r(141).concat("length","prototype");n.f=Object.getOwnPropertyNames||function(t){return e(t,i)}},function(t,n,r){var e=r(185),i=r(141);t.exports=Object.keys||function(t){return e(t,i)}},function(t,n){t.exports=function(t,n){return{enumerable:!(1&t),configurable:!(2&t),writable:!(4&t),value:n}}},function(t,n,r){var i=r(27);t.exports=function(t,n,r){for(var e in n)i(t,e,n[e],r);return t}},function(t,n,r){"use strict";var e=r(3),i=r(11),o=r(10),u=r(7)("species");t.exports=function(t){var n=e[t];o&&n&&!n[u]&&i.f(n,u,{configurable:!0,get:function(){return this}})}},function(t,n,r){var e=r(49),i=Math.max,o=Math.min;t.exports=function(t,n){return(t=e(t))<0?i(t+n,0):o(t,n)}},function(t,n){var r=0,e=Math.random();t.exports=function(t){return"Symbol(".concat(void 0===t?"":t,")_",(++r+e).toString(36))}},function(t,n,r){var e=r(5);t.exports=function(t,n){if(!e(t)||t._t!==n)throw TypeError("Incompatible receiver, "+n+" required!");return t}},function(t,n,r){var i=r(45),o=r(7)("toStringTag"),u="Arguments"==i(function(){return arguments}());t.exports=function(t){var n,r,e;return void 0===t?"Undefined":null===t?"Null":"string"==typeof(r=function(t,n){try{return t[n]}catch(t){}}(n=Object(t),o))?r:u?i(n):"Object"==(e=i(n))&&"function"==typeof n.callee?"Arguments":e}},function(t,n){t.exports={}},function(t,n,r){var e=r(11).f,i=r(30),o=r(7)("toStringTag");t.exports=function(t,n,r){t&&!i(t=r?t:t.prototype,o)&&e(t,o,{configurable:!0,value:n})}},function(t,n,r){function e(t,n,r){var e={},i=c(function(){return!!f[t]()||"
"!="
"[t]()}),o=e[t]=i?n(l):f[t];r&&(e[r]=o),u(u.P+u.F*i,"String",e)}var u=r(1),i=r(51),c=r(4),f=r(157),o="["+f+"]",a=RegExp("^"+o+o+"*"),s=RegExp(o+o+"*$"),l=e.trim=function(t,n){return t=String(i(t)),1&n&&(t=t.replace(a,"")),2&n&&(t=t.replace(s,"")),t};t.exports=e},function(t,n,r){t.exports={default:r(88),__esModule:!0}},function(t,n,r){t.exports={default:r(89),__esModule:!0}},function(t,n,r){"use strict";function e(t){return t&&t.__esModule?t:{default:t}}n.__esModule=!0;var i=e(r(86)),o=e(r(85)),u="function"==typeof o.default&&"symbol"==typeof i.default?function(t){return typeof t}:function(t){return t&&"function"==typeof o.default&&t.constructor===o.default&&t!==o.default.prototype?"symbol":typeof t};n.default="function"==typeof o.default&&"symbol"===u(i.default)?function(t){return void 0===t?"undefined":u(t)}:function(t){return t&&"function"==typeof o.default&&t.constructor===o.default&&t!==o.default.prototype?"symbol":void 0===t?"undefined":u(t)}},function(t,n,r){r(111),r(109),r(112),r(113),t.exports=r(19).Symbol},function(t,n,r){r(110),r(114),t.exports=r(44).f("iterator")},function(t,n){t.exports=function(t){if("function"!=typeof t)throw TypeError(t+" is not a function!");return t}},function(t,n){t.exports=function(){}},function(t,n,r){var f=r(15),a=r(107),s=r(106);t.exports=function(c){return function(t,n,r){var e,i=f(t),o=a(i.length),u=s(r,o);if(c&&n!=n){for(;u<o;)if((e=i[u++])!=e)return!0}else for(;u<o;u++)if((c||u in i)&&i[u]===n)return c||u||0;return!c&&-1}}},function(t,n,r){var o=r(90);t.exports=function(e,i,t){if(o(e),void 0===i)return e;switch(t){case 1:return function(t){return e.call(i,t)};case 2:return function(t,n){return e.call(i,t,n)};case 3:return function(t,n,r){return e.call(i,t,n,r)}}return function(){return e.apply(i,arguments)}}},function(t,n,r){var c=r(29),f=r(64),a=r(37);t.exports=function(t){var n=c(t),r=f.f;if(r)for(var e,i=r(t),o=a.f,u=0;i.length>u;)o.call(t,e=i[u++])&&n.push(e);return n}},function(t,n,r){var e=r(6).document;t.exports=e&&e.documentElement},function(t,n,r){var e=r(58);t.exports=Object("z").propertyIsEnumerable(0)?Object:function(t){return"String"==e(t)?t.split(""):Object(t)}},function(t,n,r){var e=r(58);t.exports=Array.isArray||function(t){return"Array"==e(t)}},function(t,n,r){"use strict";var e=r(62),i=r(24),o=r(38),u={};r(13)(u,r(16)("iterator"),function(){return this}),t.exports=function(t,n,r){t.prototype=e(u,{next:i(1,r)}),o(t,n+" Iterator")}},function(t,n){t.exports=function(t,n){return{value:n,done:!!t}}},function(t,n,r){function e(t){c(t,i,{value:{i:"O"+ ++f,w:{}}})}var i=r(25)("meta"),o=r(18),u=r(9),c=r(14).f,f=0,a=Object.isExtensible||function(){return!0},s=!r(20)(function(){return a(Object.preventExtensions({}))}),l=t.exports={KEY:i,NEED:!1,fastKey:function(t,n){if(!o(t))return"symbol"==typeof t?t:("string"==typeof t?"S":"P")+t;if(!u(t,i)){if(!a(t))return"F";if(!n)return"E";e(t)}return t[i].i},getWeak:function(t,n){if(!u(t,i)){if(!a(t))return!0;if(!n)return!1;e(t)}return t[i].w},onFreeze:function(t){return s&&l.NEED&&a(t)&&!u(t,i)&&e(t),t}}},function(t,n,r){var u=r(14),c=r(22),f=r(29);t.exports=r(12)?Object.defineProperties:function(t,n){c(t);for(var r,e=f(n),i=e.length,o=0;o<i;)u.f(t,r=e[o++],n[r]);return t}},function(t,n,r){var e=r(37),i=r(24),o=r(15),u=r(42),c=r(9),f=r(60),a=Object.getOwnPropertyDescriptor;n.f=r(12)?a:function(t,n){if(t=o(t),n=u(n,!0),f)try{return a(t,n)}catch(t){}if(c(t,n))return i(!e.f.call(t,n),t[n])}},function(t,n,r){var e=r(15),i=r(63).f,o={}.toString,u="object"==typeof window&&window&&Object.getOwnPropertyNames?Object.getOwnPropertyNames(window):[];t.exports.f=function(t){return u&&"[object Window]"==o.call(t)?function(t){try{return i(t)}catch(t){return u.slice()}}(t):i(e(t))}},function(t,n,r){var e=r(9),i=r(55),o=r(39)("IE_PROTO"),u=Object.prototype;t.exports=Object.getPrototypeOf||function(t){return t=i(t),e(t,o)?t[o]:"function"==typeof t.constructor&&t instanceof t.constructor?t.constructor.prototype:t instanceof Object?u:null}},function(t,n,r){var f=r(41),a=r(34);t.exports=function(c){return function(t,n){var r,e,i=String(a(t)),o=f(n),u=i.length;return o<0||u<=o?c?"":void 0:(r=i.charCodeAt(o))<55296||56319<r||o+1===u||(e=i.charCodeAt(o+1))<56320||57343<e?c?i.charAt(o):r:c?i.slice(o,o+2):e-56320+(r-55296<<10)+65536}}},function(t,n,r){var e=r(41),i=Math.max,o=Math.min;t.exports=function(t,n){return(t=e(t))<0?i(t+n,0):o(t,n)}},function(t,n,r){var e=r(41),i=Math.min;t.exports=function(t){return 0<t?i(e(t),9007199254740991):0}},function(t,n,r){"use strict";var e=r(91),i=r(99),o=r(36),u=r(15);t.exports=r(61)(Array,"Array",function(t,n){this._t=u(t),this._i=0,this._k=n},function(){var t=this._t,n=this._k,r=this._i++;return!t||r>=t.length?(this._t=void 0,i(1)):i(0,"keys"==n?r:"values"==n?t[r]:[r,t[r]])},"values"),o.Arguments=o.Array,e("keys"),e("values"),e("entries")},function(t,n){},function(t,n,r){"use strict";var e=r(105)(!0);r(61)(String,"String",function(t){this._t=String(t),this._i=0},function(){var t,n=this._t,r=this._i;return r>=n.length?{value:void 0,done:!0}:(t=e(n,r),this._i+=t.length,{value:t,done:!1})})},function(t,n,r){"use strict";function e(t){var n=H[t]=A(G[V]);return n._k=t,n}function i(t,n){O(t);for(var r,e=w(n=P(n)),i=0,o=e.length;i<o;)nt(t,r=e[i++],n[r]);return t}function o(t){var n=z.call(this,t=j(t,!0));return!(this===$&&s(H,t)&&!s(J,t))&&(!(n||!s(this,t)||!s(H,t)||s(this,B)&&this[B][t])||n)}function u(t,n){if(t=P(t),n=j(n,!0),t!==$||!s(H,n)||s(J,n)){var r=R(t,n);return!r||!s(H,n)||s(t,B)&&t[B][n]||(r.enumerable=!0),r}}function c(t){for(var n,r=D(P(t)),e=[],i=0;r.length>i;)s(H,n=r[i++])||n==B||n==p||e.push(n);return e}function f(t){for(var n,r=t===$,e=D(r?J:P(t)),i=[],o=0;e.length>o;)!s(H,n=e[o++])||r&&!s($,n)||i.push(H[n]);return i}var a=r(6),s=r(9),l=r(12),h=r(54),v=r(66),p=r(100).KEY,d=r(20),y=r(40),g=r(38),b=r(25),x=r(16),m=r(44),S=r(43),w=r(94),_=r(97),O=r(22),E=r(18),M=r(55),P=r(15),j=r(42),F=r(24),A=r(62),I=r(103),L=r(102),N=r(64),T=r(14),k=r(29),R=L.f,C=T.f,D=I.f,G=a.Symbol,W=a.JSON,U=W&&W.stringify,V="prototype",B=x("_hidden"),q=x("toPrimitive"),z={}.propertyIsEnumerable,K=y("symbol-registry"),H=y("symbols"),J=y("op-symbols"),$=Object[V],Y="function"==typeof G&&!!N.f,X=a.QObject,Q=!X||!X[V]||!X[V].findChild,Z=l&&d(function(){return 7!=A(C({},"a",{get:function(){return C(this,"a",{value:7}).a}})).a})?function(t,n,r){var e=R($,n);e&&delete $[n],C(t,n,r),e&&t!==$&&C($,n,e)}:C,tt=Y&&"symbol"==typeof G.iterator?function(t){return"symbol"==typeof t}:function(t){return t instanceof G},nt=function(t,n,r){return t===$&&nt(J,n,r),O(t),n=j(n,!0),O(r),s(H,n)?(r.enumerable?(s(t,B)&&t[B][n]&&(t[B][n]=!1),r=A(r,{enumerable:F(0,!1)})):(s(t,B)||C(t,B,F(1,{})),t[B][n]=!0),Z(t,n,r)):C(t,n,r)};Y||(v((G=function(){if(this instanceof G)throw TypeError("Symbol is not a constructor!");var n=b(0<arguments.length?arguments[0]:void 0),r=function(t){this===$&&r.call(J,t),s(this,B)&&s(this[B],n)&&(this[B][n]=!1),Z(this,n,F(1,t))};return l&&Q&&Z($,n,{configurable:!0,set:r}),e(n)})[V],"toString",function(){return this._k}),L.f=u,T.f=nt,r(63).f=I.f=c,r(37).f=o,N.f=f,l&&!r(23)&&v($,"propertyIsEnumerable",o,!0),m.f=function(t){return e(x(t))}),h(h.G+h.W+h.F*!Y,{Symbol:G});for(var rt="hasInstance,isConcatSpreadable,iterator,match,replace,search,species,split,toPrimitive,toStringTag,unscopables".split(","),et=0;rt.length>et;)x(rt[et++]);for(var it=k(x.store),ot=0;it.length>ot;)S(it[ot++]);h(h.S+h.F*!Y,"Symbol",{for:function(t){return s(K,t+="")?K[t]:K[t]=G(t)},keyFor:function(t){if(!tt(t))throw TypeError(t+" is not a symbol!");for(var n in K)if(K[n]===t)return n},useSetter:function(){Q=!0},useSimple:function(){Q=!1}}),h(h.S+h.F*!Y,"Object",{create:function(t,n){return void 0===n?A(t):i(A(t),n)},defineProperty:nt,defineProperties:i,getOwnPropertyDescriptor:u,getOwnPropertyNames:c,getOwnPropertySymbols:f});var ut=d(function(){N.f(1)});h(h.S+h.F*ut,"Object",{getOwnPropertySymbols:function(t){return N.f(M(t))}}),W&&h(h.S+h.F*(!Y||d(function(){var t=G();return"[null]"!=U([t])||"{}"!=U({a:t})||"{}"!=U(Object(t))})),"JSON",{stringify:function(t){for(var n,r,e=[t],i=1;i<arguments.length;)e.push(arguments[i++]);if(r=n=e[1],(E(n)||void 0!==t)&&!tt(t))return _(n)||(n=function(t,n){if("function"==typeof r&&(n=r.call(this,t,n)),!tt(n))return n}),e[1]=n,U.apply(W,e)}}),G[V][q]||r(13)(G[V],q,G[V].valueOf),g(G,"Symbol"),g(Math,"Math",!0),g(a.JSON,"JSON",!0)},function(t,n,r){r(43)("asyncIterator")},function(t,n,r){r(43)("observable")},function(t,n,r){r(108);for(var e=r(6),i=r(13),o=r(36),u=r(16)("toStringTag"),c="CSSRuleList,CSSStyleDeclaration,CSSValueList,ClientRectList,DOMRectList,DOMStringList,DOMTokenList,DataTransferItemList,FileList,HTMLAllCollection,HTMLCollection,HTMLFormElement,HTMLSelectElement,MediaList,MimeTypeArray,NamedNodeMap,NodeList,PaintRequestList,Plugin,PluginArray,SVGLengthList,SVGNumberList,SVGPathSegList,SVGPointList,SVGStringList,SVGTransformList,SourceBufferList,StyleSheetList,TextTrackCueList,TextTrackList,TouchList".split(","),f=0;f<c.length;f++){var a=c[f],s=e[a],l=s&&s.prototype;l&&!l[u]&&i(l,u,a),o[a]=o.Array}},function(t,n,r){"use strict";var e=r(2);t.exports=function(){var t=e(this),n="";return t.global&&(n+="g"),t.ignoreCase&&(n+="i"),t.multiline&&(n+="m"),t.unicode&&(n+="u"),t.sticky&&(n+="y"),n}},function(t,n,r){var e=r(45);t.exports=Object("z").propertyIsEnumerable(0)?Object:function(t){return"String"==e(t)?t.split(""):Object(t)}},function(t,n){n.f={}.propertyIsEnumerable},function(t,n,r){var e=r(46),i=r(3),o="__core-js_shared__",u=i[o]||(i[o]={});(t.exports=function(t,n){return u[t]||(u[t]=void 0!==n?n:{})})("versions",[]).push({version:e.version,mode:r(68)?"pure":"global",copyright:"© 2019 Denis Pushkarev (zloirock.ru)"})},function(t,n,r){var i=r(2),o=r(21),u=r(7)("species");t.exports=function(t,n){var r,e=i(t).constructor;return void 0===e||null==(r=i(e)[u])?n:o(r)}},function(t,n,r){var f=r(33),a=r(8),s=r(78);t.exports=function(c){return function(t,n,r){var e,i=f(t),o=a(i.length),u=s(r,o);if(c&&n!=n){for(;u<o;)if((e=i[u++])!=e)return!0}else for(;u<o;u++)if((c||u in i)&&i[u]===n)return c||u||0;return!c&&-1}}},function(t,n,r){"use strict";var g=r(3),b=r(1),x=r(27),m=r(76),S=r(69),w=r(71),_=r(70),O=r(5),E=r(4),M=r(125),P=r(83),j=r(144);t.exports=function(e,t,n,r,i,o){function u(t){var r=d[t];x(d,t,"delete"==t?function(t){return!(o&&!O(t))&&r.call(this,0===t?0:t)}:"has"==t?function(t){return!(o&&!O(t))&&r.call(this,0===t?0:t)}:"get"==t?function(t){return o&&!O(t)?void 0:r.call(this,0===t?0:t)}:"add"==t?function(t){return r.call(this,0===t?0:t),this}:function(t,n){return r.call(this,0===t?0:t,n),this})}var c,f,a,s,l,h=g[e],v=h,p=i?"set":"add",d=v&&v.prototype,y={};return"function"==typeof v&&(o||d.forEach&&!E(function(){(new v).entries().next()}))?(f=(c=new v)[p](o?{}:-0,1)!=c,a=E(function(){c.has(1)}),s=M(function(t){new v(t)}),l=!o&&E(function(){for(var t=new v,n=5;n--;)t[p](n,n);return!t.has(-0)}),s||(((v=t(function(t,n){_(t,v,e);var r=j(new h,t,v);return null!=n&&w(n,i,r[p],r),r})).prototype=d).constructor=v),(a||l)&&(u("delete"),u("has"),i&&u("get")),(l||f)&&u(p),o&&d.clear&&delete d.clear):(v=r.getConstructor(t,e,i,p),m(v.prototype,n),S.NEED=!0),P(v,e),y[e]=v,b(b.G+b.W+b.F*(v!=h),y),o||r.setStrong(v,e,i),v}},function(t,n,r){"use strict";r(197);var s=r(27),l=r(26),h=r(4),v=r(51),p=r(7),d=r(152),y=p("species"),g=!h(function(){var t=/./;return t.exec=function(){var t=[];return t.groups={a:"7"},t},"7"!=="".replace(t,"$<a>")}),b=function(){var t=/(?:)/,n=t.exec;t.exec=function(){return n.apply(this,arguments)};var r="ab".split(t);return 2===r.length&&"a"===r[0]&&"b"===r[1]}();t.exports=function(r,t,n){var o,e,i,u,c=p(r),f=!h(function(){var t={};return t[c]=function(){return 7},7!=""[r](t)}),a=f?!h(function(){var t=!1,n=/a/;return n.exec=function(){return t=!0,null},"split"===r&&(n.constructor={},n.constructor[y]=function(){return n}),n[c](""),!t}):void 0;f&&a&&("replace"!==r||g)&&("split"!==r||b)||(o=/./[c],i=(e=n(v,c,""[r],function(t,n,r,e,i){return n.exec===d?f&&!i?{done:!0,value:o.call(n,r,e)}:{done:!0,value:t.call(r,n,e)}:{done:!1}}))[0],u=e[1],s(String.prototype,r,i),l(RegExp.prototype,c,2==t?function(t,n){return u.call(t,this,n)}:function(t){return u.call(t,this)}))}},function(t,n,r){var e=r(45);t.exports=Array.isArray||function(t){return"Array"==e(t)}},function(t,n,r){var e=r(5),i=r(45),o=r(7)("match");t.exports=function(t){var n;return e(t)&&(void 0!==(n=t[o])?!!n:"RegExp"==i(t))}},function(t,n,r){var o=r(7)("iterator"),u=!1;try{var e=[7][o]();e.return=function(){u=!0},Array.from(e,function(){throw 2})}catch(t){}t.exports=function(t,n){if(!n&&!u)return!1;var r=!1;try{var e=[7],i=e[o]();i.next=function(){return{done:r=!0}},e[o]=function(){return i},t(e)}catch(t){}return r}},function(t,n,r){"use strict";t.exports=r(68)||!r(4)(function(){var t=Math.random();__defineSetter__.call(null,t,function(){}),delete r(3)[t]})},function(t,n){n.f=Object.getOwnPropertySymbols},function(t,n,r){"use strict";var i=r(81),o=RegExp.prototype.exec;t.exports=function(t,n){var r=t.exec;if("function"==typeof r){var e=r.call(t,n);if("object"!=typeof e)throw new TypeError("RegExp exec method returned something other than an Object or null");return e}if("RegExp"!==i(t))throw new TypeError("RegExp#exec called on incompatible receiver");return o.call(t,n)}},function(t,n,r){"use strict";var e=r(1),u=r(21),c=r(47),f=r(71);t.exports=function(t){e(e.S,t,{from:function(t){var n,r,e,i,o=arguments[1];return u(this),(n=void 0!==o)&&u(o),null==t?new this:(r=[],n?(e=0,i=c(o,arguments[2],2),f(t,!1,function(t){r.push(i(t,e++))})):f(t,!1,r.push,r),new this(r))}})}},function(t,n,r){"use strict";var e=r(1);t.exports=function(t){e(e.S,t,{of:function(){for(var t=arguments.length,n=new Array(t);t--;)n[t]=arguments[t];return new this(n)}})}},function(t,n,r){var f=r(49),a=r(51);t.exports=function(c){return function(t,n){var r,e,i=String(a(t)),o=f(n),u=i.length;return o<0||u<=o?c?"":void 0:(r=i.charCodeAt(o))<55296||56319<r||o+1===u||(e=i.charCodeAt(o+1))<56320||57343<e?c?i.charAt(o):r:c?i.slice(o,o+2):e-56320+(r-55296<<10)+65536}}},function(t,n,r){for(var e,i=r(3),o=r(26),u=r(79),c=u("typed_array"),f=u("view"),a=!(!i.ArrayBuffer||!i.DataView),s=a,l=0,h="Int8Array,Uint8Array,Uint8ClampedArray,Int16Array,Uint16Array,Int32Array,Uint32Array,Float32Array,Float64Array".split(",");l<9;)(e=i[h[l++]])?(o(e.prototype,c,!0),o(e.prototype,f,!0)):s=!1;t.exports={ABV:a,CONSTR:s,TYPED:c,VIEW:f}},function(t,n,r){var e=r(3).navigator;t.exports=e&&e.userAgent||""},function(t,n){"use strict";var r,e={versions:{trident:-1<(r=window.navigator.userAgent).indexOf("Trident"),presto:-1<r.indexOf("Presto"),webKit:-1<r.indexOf("AppleWebKit"),gecko:-1<r.indexOf("Gecko")&&-1==r.indexOf("KHTML"),mobile:!!r.match(/AppleWebKit.*Mobile.*/),ios:!!r.match(/\(i[^;]+;( U;)? CPU.+Mac OS X/),android:-1<r.indexOf("Android")||-1<r.indexOf("Linux"),iPhone:-1<r.indexOf("iPhone")||-1<r.indexOf("Mac"),iPad:-1<r.indexOf("iPad"),webApp:-1==r.indexOf("Safari"),weixin:-1==r.indexOf("MicroMessenger")}};t.exports=e},function(t,n,r){"use strict";var e=r(87),l=e&&e.__esModule?e:{default:e},h=function(){function n(t,n,r){return n||r?String.fromCharCode(n||r):o[t]||t}function r(t){return s[t]}var e=/"|<|>|&| |'|&#(\d+);|&#(\d+)/g,i=/['<> "&]/g,o={""":'"',"<":"<",">":">","&":"&"," ":" "},u=/\u00a0/g,c=/<br\s*\/?>/gi,f=/\r?\n/g,a=/\s/g,s={};for(var t in o)s[o[t]]=t;return o["'"]="'",s["'"]="'",{encode:function(t){return t?(""+t).replace(i,r).replace(f,"<br/>").replace(a," "):""},decode:function(t){return t?(""+t).replace(c,"\n").replace(e,n).replace(u," "):""},encodeBase16:function(t){if(!t)return t;for(var n=[],r=0,e=(t+="").length;r<e;r++)n.push(t.charCodeAt(r).toString(16).toUpperCase());return n.join("")},encodeBase16forJSON:function(t){if(!t)return t;for(var n=[],r=0,e=(t=t.replace(/[\u4E00-\u9FBF]/gi,function(t){return escape(t).replace("%u","\\u")})).length;r<e;r++)n.push(t.charCodeAt(r).toString(16).toUpperCase());return n.join("")},decodeBase16:function(t){if(!t)return t;for(var n=[],r=0,e=(t+="").length;r<e;r+=2)n.push(String.fromCharCode("0x"+t.slice(r,r+2)));return n.join("")},encodeObject:function(t){if(t instanceof Array)for(var n=0,r=t.length;n<r;n++)t[n]=h.encodeObject(t[n]);else if("object"==(void 0===t?"undefined":(0,l.default)(t)))for(var e in t)t[e]=h.encodeObject(t[e]);else if("string"==typeof t)return h.encode(t);return t},loadScript:function(t){var n=document.createElement("script");document.getElementsByTagName("body")[0].appendChild(n),n.setAttribute("src",t)},addLoadEvent:function(t){var n=window.onload;"function"!=typeof window.onload?window.onload=t:window.onload=function(){n(),t()}}}}();t.exports=h},function(t,n,r){"use strict";var e=r(131)(!0);t.exports=function(t,n,r){return n+(r?e(t,n).length:1)}},function(t,n,r){"use strict";var c=r(17),f=r(78),a=r(8);t.exports=function(t){for(var n=c(this),r=a(n.length),e=arguments.length,i=f(1<e?arguments[1]:void 0,r),o=2<e?arguments[2]:void 0,u=void 0===o?r:f(o,r);i<u;)n[i++]=t;return n}},function(t,n,r){var e=r(215);t.exports=function(t,n){return new(e(t))(n)}},function(t,n,r){"use strict";var e=r(11),i=r(75);t.exports=function(t,n,r){n in t?e.f(t,n,i(0,r)):t[n]=r}},function(t,n,r){var e=r(5),i=r(3).document,o=e(i)&&e(i.createElement);t.exports=function(t){return o?i.createElement(t):{}}},function(t,n){t.exports="constructor,hasOwnProperty,isPrototypeOf,propertyIsEnumerable,toLocaleString,toString,valueOf".split(",")},function(t,n,r){var e=r(7)("match");t.exports=function(n){var r=/./;try{"/./"[n](r)}catch(t){try{return r[e]=!1,!"/./"[n](r)}catch(n){}}return!0}},function(t,n,r){var e=r(3).document;t.exports=e&&e.documentElement},function(t,n,r){var o=r(5),u=r(153).set;t.exports=function(t,n,r){var e,i=n.constructor;return i!==r&&"function"==typeof i&&(e=i.prototype)!==r.prototype&&o(e)&&u&&u(t,e),t}},function(t,n,r){var e=r(82),i=r(7)("iterator"),o=Array.prototype;t.exports=function(t){return void 0!==t&&(e.Array===t||o[i]===t)}},function(t,n,r){"use strict";var e=r(72),i=r(75),o=r(83),u={};r(26)(u,r(7)("iterator"),function(){return this}),t.exports=function(t,n,r){t.prototype=e(u,{next:i(1,r)}),o(t,n+" Iterator")}},function(t,n,r){"use strict";function x(){return this}var m=r(68),S=r(1),w=r(27),_=r(26),O=r(82),E=r(146),M=r(83),P=r(32),j=r(7)("iterator"),F=!([].keys&&"next"in[].keys()),A="values";t.exports=function(t,n,r,e,i,o,u){function c(t){if(!F&&t in p)return p[t];switch(t){case"keys":case A:return function(){return new r(this,t)}}return function(){return new r(this,t)}}E(r,n,e);var f,a,s,l=n+" Iterator",h=i==A,v=!1,p=t.prototype,d=p[j]||p["@@iterator"]||i&&p[i],y=d||c(i),g=i?h?c("entries"):y:void 0,b="Array"==n&&p.entries||d;if(b&&(s=P(b.call(new t)))!==Object.prototype&&s.next&&(M(s,l,!0),m||"function"==typeof s[j]||_(s,j,x)),h&&d&&d.name!==A&&(v=!0,y=function(){return d.call(this)}),m&&!u||!F&&!v&&p[j]||_(p,j,y),O[n]=y,O[l]=x,i)if(f={values:h?y:c(A),keys:o?y:c("keys"),entries:g},u)for(a in f)a in p||w(p,a,f[a]);else S(S.P+S.F*(F||v),n,f);return f}},function(t,n){var r=Math.expm1;t.exports=!r||22025.465794806718<r(10)||r(10)<22025.465794806718||-2e-17!=r(-2e-17)?function(t){return 0==(t=+t)?t:-1e-6<t&&t<1e-6?t+t*t/2:Math.exp(t)-1}:r},function(t,n){t.exports=Math.sign||function(t){return 0==(t=+t)||t!=t?t:t<0?-1:1}},function(t,n,r){var c=r(3),f=r(158).set,a=c.MutationObserver||c.WebKitMutationObserver,s=c.process,l=c.Promise,h="process"==r(45)(s);t.exports=function(){function t(){var t,n;for(h&&(t=s.domain)&&t.exit();r;){n=r.fn,r=r.next;try{n()}catch(t){throw r?u():e=void 0,t}}e=void 0,t&&t.enter()}var r,e,n,i,o,u=h?function(){s.nextTick(t)}:!a||c.navigator&&c.navigator.standalone?l&&l.resolve?(n=l.resolve(void 0),function(){n.then(t)}):function(){f.call(c,t)}:(i=!0,o=document.createTextNode(""),new a(t).observe(o,{characterData:!0}),function(){o.data=i=!i});return function(t){var n={fn:t,next:void 0};e&&(e.next=n),r||(r=n,u()),e=n}}},function(t,n,r){"use strict";function e(t){var r,e;this.promise=new t(function(t,n){if(void 0!==r||void 0!==e)throw TypeError("Bad Promise constructor");r=t,e=n}),this.resolve=i(r),this.reject=i(e)}var i=r(21);t.exports.f=function(t){return new e(t)}},function(t,n,r){"use strict";var e,i,u=r(115),c=RegExp.prototype.exec,f=String.prototype.replace,o=c,a="lastIndex",s=(e=/a/,i=/b*/g,c.call(e,"a"),c.call(i,"a"),0!==e[a]||0!==i[a]),l=void 0!==/()??/.exec("")[1];(s||l)&&(o=function(t){var n,r,e,i,o=this;return l&&(r=new RegExp("^"+o.source+"$(?!\\s)",u.call(o))),s&&(n=o[a]),e=c.call(o,t),s&&e&&(o[a]=o.global?e.index+e[0].length:n),l&&e&&1<e.length&&f.call(e[0],r,function(){for(i=1;i<arguments.length-2;i++)void 0===arguments[i]&&(e[i]=void 0)}),e}),t.exports=o},function(t,n,i){function o(t,n){if(e(t),!r(n)&&null!==n)throw TypeError(n+": can't set as prototype!")}var r=i(5),e=i(2);t.exports={set:Object.setPrototypeOf||("__proto__"in{}?function(t,r,e){try{(e=i(47)(Function.call,i(31).f(Object.prototype,"__proto__").set,2))(t,[]),r=!(t instanceof Array)}catch(t){r=!0}return function(t,n){return o(t,n),r?t.__proto__=n:e(t,n),t}}({},!1):void 0),check:o}},function(t,n,r){var e=r(118)("keys"),i=r(79);t.exports=function(t){return e[t]||(e[t]=i(t))}},function(t,n,r){var e=r(124),i=r(51);t.exports=function(t,n,r){if(e(n))throw TypeError("String#"+r+" doesn't accept regex!");return String(i(t))}},function(t,n,r){"use strict";var i=r(49),o=r(51);t.exports=function(t){var n=String(o(this)),r="",e=i(t);if(e<0||e==1/0)throw RangeError("Count can't be negative");for(;0<e;(e>>>=1)&&(n+=n))1&e&&(r+=n);return r}},function(t,n){t.exports="\t\n\v\f\r \u2028\u2029\ufeff"},function(t,n,r){function e(){var t,n=+this;x.hasOwnProperty(n)&&(t=x[n],delete x[n],t())}function i(t){e.call(t.data)}var o,u,c,f=r(47),a=r(175),s=r(143),l=r(140),h=r(3),v=h.process,p=h.setImmediate,d=h.clearImmediate,y=h.MessageChannel,g=h.Dispatch,b=0,x={},m="onreadystatechange";p&&d||(p=function(t){for(var n=[],r=1;r<arguments.length;)n.push(arguments[r++]);return x[++b]=function(){a("function"==typeof t?t:Function(t),n)},o(b),b},d=function(t){delete x[t]},"process"==r(45)(v)?o=function(t){v.nextTick(f(e,t,1))}:g&&g.now?o=function(t){g.now(f(e,t,1))}:y?(c=(u=new y).port2,u.port1.onmessage=i,o=f(c.postMessage,c,1)):h.addEventListener&&"function"==typeof postMessage&&!h.importScripts?(o=function(t){h.postMessage(t+"","*")},h.addEventListener("message",i,!1)):o=m in l("script")?function(t){s.appendChild(l("script"))[m]=function(){s.removeChild(this),e.call(t)}}:function(t){setTimeout(f(e,t,1),0)}),t.exports={set:p,clear:d}},function(t,n,r){"use strict";function e(t,n,r){var e,i,o,u=new Array(r),c=8*r-n-1,f=(1<<c)-1,a=f>>1,s=23===n?W(2,-24)-W(2,-77):0,l=0,h=t<0||0===t&&1/t<0?1:0;for((t=G(t))!=t||t===C?(i=t!=t?1:0,e=f):(e=U(V(t)/B),t*(o=W(2,-e))<1&&(e--,o*=2),2<=(t+=1<=e+a?s/o:s*W(2,1-a))*o&&(e++,o/=2),f<=e+a?(i=0,e=f):1<=e+a?(i=(t*o-1)*W(2,n),e+=a):(i=t*W(2,a-1)*W(2,n),e=0));8<=n;u[l++]=255&i,i/=256,n-=8);for(e=e<<n|i,c+=n;0<c;u[l++]=255&e,e/=256,c-=8);return u[--l]|=128*h,u}function i(t,n,r){var e,i=8*r-n-1,o=(1<<i)-1,u=o>>1,c=i-7,f=r-1,a=t[f--],s=127&a;for(a>>=7;0<c;s=256*s+t[f],f--,c-=8);for(e=s&(1<<-c)-1,s>>=-c,c+=n;0<c;e=256*e+t[f],f--,c-=8);if(0===s)s=1-u;else{if(s===o)return e?NaN:a?-C:C;e+=W(2,n),s-=u}return(a?-1:1)*e*W(2,s-n)}function o(t){return t[3]<<24|t[2]<<16|t[1]<<8|t[0]}function u(t){return[255&t]}function c(t){return[255&t,t>>8&255]}function f(t){return[255&t,t>>8&255,t>>16&255,t>>24&255]}function a(t){return e(t,52,8)}function s(t){return e(t,23,4)}function l(t,n,r){M(t[I],n,{get:function(){return this[r]}})}function h(t,n,r,e){var i=O(+r);if(i+n>t[H])throw R(L);var o=t[K]._b,u=i+t[J],c=o.slice(u,u+n);return e?c:c.reverse()}function v(t,n,r,e,i,o){var u=O(+r);if(u+n>t[H])throw R(L);for(var c=t[K]._b,f=u+t[J],a=e(+i),s=0;s<n;s++)c[f+s]=a[o?s:n-s-1]}var p=r(3),d=r(10),y=r(68),g=r(132),b=r(26),x=r(76),m=r(4),S=r(70),w=r(49),_=r(8),O=r(194),E=r(73).f,M=r(11).f,P=r(137),j=r(83),F="ArrayBuffer",A="DataView",I="prototype",L="Wrong index!",N=p[F],T=p[A],k=p.Math,R=p.RangeError,C=p.Infinity,D=N,G=k.abs,W=k.pow,U=k.floor,V=k.log,B=k.LN2,q="byteLength",z="byteOffset",K=d?"_b":"buffer",H=d?"_l":q,J=d?"_o":z;if(g.ABV){if(!m(function(){N(1)})||!m(function(){new N(-1)})||m(function(){return new N,new N(1.5),new N(NaN),N.name!=F})){for(var $,Y=(N=function(t){return S(this,N),new D(O(t))})[I]=D[I],X=E(D),Q=0;X.length>Q;)($=X[Q++])in N||b(N,$,D[$]);y||(Y.constructor=N)}var Z=new T(new N(2)),tt=T[I].setInt8;Z.setInt8(0,2147483648),Z.setInt8(1,2147483649),!Z.getInt8(0)&&Z.getInt8(1)||x(T[I],{setInt8:function(t,n){tt.call(this,t,n<<24>>24)},setUint8:function(t,n){tt.call(this,t,n<<24>>24)}},!0)}else N=function(t){S(this,N,F);var n=O(t);this._b=P.call(new Array(n),0),this[H]=n},T=function(t,n,r){S(this,T,A),S(t,N,A);var e=t[H],i=w(n);if(i<0||e<i)throw R("Wrong offset!");if(e<i+(r=void 0===r?e-i:_(r)))throw R("Wrong length!");this[K]=t,this[J]=i,this[H]=r},d&&(l(N,q,"_l"),l(T,"buffer","_b"),l(T,q,"_l"),l(T,z,"_o")),x(T[I],{getInt8:function(t){return h(this,1,t)[0]<<24>>24},getUint8:function(t){return h(this,1,t)[0]},getInt16:function(t){var n=h(this,2,t,arguments[1]);return(n[1]<<8|n[0])<<16>>16},getUint16:function(t){var n=h(this,2,t,arguments[1]);return n[1]<<8|n[0]},getInt32:function(t){return o(h(this,4,t,arguments[1]))},getUint32:function(t){return o(h(this,4,t,arguments[1]))>>>0},getFloat32:function(t){return i(h(this,4,t,arguments[1]),23,4)},getFloat64:function(t){return i(h(this,8,t,arguments[1]),52,8)},setInt8:function(t,n){v(this,1,t,u,n)},setUint8:function(t,n){v(this,1,t,u,n)},setInt16:function(t,n){v(this,2,t,c,n,arguments[2])},setUint16:function(t,n){v(this,2,t,c,n,arguments[2])},setInt32:function(t,n){v(this,4,t,f,n,arguments[2])},setUint32:function(t,n){v(this,4,t,f,n,arguments[2])},setFloat32:function(t,n){v(this,4,t,s,n,arguments[2])},setFloat64:function(t,n){v(this,8,t,a,n,arguments[2])}});j(N,F),j(T,A),b(T[I],g.VIEW,!0),n[F]=N,n[A]=T},function(t,n,r){var e=r(3),i=r(46),o=r(68),u=r(195),c=r(11).f;t.exports=function(t){var n=i.Symbol||(i.Symbol=!o&&e.Symbol||{});"_"==t.charAt(0)||t in n||c(n,t,{value:u.f(t)})}},function(t,n,r){var e=r(81),i=r(7)("iterator"),o=r(82);t.exports=r(46).getIteratorMethod=function(t){if(null!=t)return t[i]||t["@@iterator"]||o[e(t)]}},function(t,n,r){"use strict";var e=r(67),i=r(178),o=r(82),u=r(33);t.exports=r(147)(Array,"Array",function(t,n){this._t=u(t),this._i=0,this._k=n},function(){var t=this._t,n=this._k,r=this._i++;return!t||r>=t.length?(this._t=void 0,i(1)):i(0,"keys"==n?r:"values"==n?t[r]:[r,t[r]])},"values"),o.Arguments=o.Array,e("keys"),e("values"),e("entries")},function(t,n){t.exports=function(t,n){t.classList?t.classList.add(n):t.className+=" "+n}},function(t,n){t.exports=function(t,n){var r;t.classList?t.classList.remove(n):(r=new RegExp("(^|\\b)"+n.split(" ").join("|")+"(\\b|$)","gi"),t.className=t.className.replace(r," "))}},function(t,n,r){var e=r(45);t.exports=function(t,n){if("number"!=typeof t&&"Number"!=e(t))throw TypeError(n);return+t}},function(t,n,r){"use strict";var a=r(17),s=r(78),l=r(8);t.exports=[].copyWithin||function(t,n){var r=a(this),e=l(r.length),i=s(t,e),o=s(n,e),u=2<arguments.length?arguments[2]:void 0,c=Math.min((void 0===u?e:s(u,e))-o,e-i),f=1;for(o<i&&i<o+c&&(f=-1,o+=c-1,i+=c-1);0<c--;)o in r?r[i]=r[o]:delete r[i],i+=f,o+=f;return r}},function(t,n,r){var e=r(71);t.exports=function(t,n){var r=[];return e(t,!1,r.push,r,n),r}},function(t,n,r){var s=r(21),l=r(17),h=r(116),v=r(8);t.exports=function(t,n,r,e,i){s(n);var o=l(t),u=h(o),c=v(o.length),f=i?c-1:0,a=i?-1:1;if(r<2)for(;;){if(f in u){e=u[f],f+=a;break}if(f+=a,i?f<0:c<=f)throw TypeError("Reduce of empty array with no initial value")}for(;i?0<=f:f<c;f+=a)f in u&&(e=n(e,u[f],f,o));return e}},function(t,n,r){"use strict";var o=r(21),u=r(5),c=r(175),f=[].slice,a={};t.exports=Function.bind||function(n){var r=o(this),e=f.call(arguments,1),i=function(){var t=e.concat(f.call(arguments));return this instanceof i?function(t,n,r){if(!(n in a)){for(var e=[],i=0;i<n;i++)e[i]="a["+i+"]";a[n]=Function("F,a","return new F("+e.join(",")+")")}return a[n](t,r)}(r,t.length,t):c(r,t,n)};return u(r.prototype)&&(i.prototype=r.prototype),i}},function(t,n,r){"use strict";function u(t,n){var r,e=p(n);if("F"!==e)return t._i[e];for(r=t._f;r;r=r.n)if(r.k==n)return r}var c=r(11).f,f=r(72),a=r(76),s=r(47),l=r(70),h=r(71),e=r(147),i=r(178),o=r(77),v=r(10),p=r(69).fastKey,d=r(80),y=v?"_s":"size";t.exports={getConstructor:function(t,o,r,e){var i=t(function(t,n){l(t,i,o,"_i"),t._t=o,t._i=f(null),t._f=void 0,t._l=void 0,t[y]=0,null!=n&&h(n,r,t[e],t)});return a(i.prototype,{clear:function(){for(var t=d(this,o),n=t._i,r=t._f;r;r=r.n)r.r=!0,r.p&&(r.p=r.p.n=void 0),delete n[r.i];t._f=t._l=void 0,t[y]=0},delete:function(t){var n,r,e=d(this,o),i=u(e,t);return i&&(n=i.n,r=i.p,delete e._i[i.i],i.r=!0,r&&(r.n=n),n&&(n.p=r),e._f==i&&(e._f=n),e._l==i&&(e._l=r),e[y]--),!!i},forEach:function(t){d(this,o);for(var n,r=s(t,1<arguments.length?arguments[1]:void 0,3);n=n?n.n:this._f;)for(r(n.v,n.k,this);n&&n.r;)n=n.p},has:function(t){return!!u(d(this,o),t)}}),v&&c(i.prototype,"size",{get:function(){return d(this,o)[y]}}),i},def:function(t,n,r){var e,i,o=u(t,n);return o?o.v=r:(t._l=o={i:i=p(n,!0),k:n,v:r,p:e=t._l,n:void 0,r:!1},t._f||(t._f=o),e&&(e.n=o),t[y]++,"F"!==i&&(t._i[i]=o)),t},getEntry:u,setStrong:function(t,r,n){e(t,r,function(t,n){this._t=d(t,r),this._k=n,this._l=void 0},function(){for(var t=this,n=t._k,r=t._l;r&&r.r;)r=r.p;return t._t&&(t._l=r=r?r.n:t._t._f)?i(0,"keys"==n?r.k:"values"==n?r.v:[r.k,r.v]):(t._t=void 0,i(1))},n?"entries":"values",!n,!0),o(r)}}},function(t,n,r){var e=r(81),i=r(167);t.exports=function(t){return function(){if(e(this)!=t)throw TypeError(t+"#toJSON isn't generic");return i(this)}}},function(t,n,r){"use strict";function u(t){return t._l||(t._l=new i)}function e(t,n){return d(t.a,function(t){return t[0]===n})}function i(){this.a=[]}var c=r(76),f=r(69).getWeak,o=r(2),a=r(5),s=r(70),l=r(71),h=r(50),v=r(30),p=r(80),d=h(5),y=h(6),g=0;i.prototype={get:function(t){var n=e(this,t);if(n)return n[1]},has:function(t){return!!e(this,t)},set:function(t,n){var r=e(this,t);r?r[1]=n:this.a.push([t,n])},delete:function(n){var t=y(this.a,function(t){return t[0]===n});return~t&&this.a.splice(t,1),!!~t}},t.exports={getConstructor:function(t,r,e,i){var o=t(function(t,n){s(t,o,r,"_i"),t._t=r,t._i=g++,t._l=void 0,null!=n&&l(n,e,t[i],t)});return c(o.prototype,{delete:function(t){if(!a(t))return!1;var n=f(t);return!0===n?u(p(this,r)).delete(t):n&&v(n,this._i)&&delete n[this._i]},has:function(t){if(!a(t))return!1;var n=f(t);return!0===n?u(p(this,r)).has(t):n&&v(n,this._i)}}),o},def:function(t,n,r){var e=f(o(n),!0);return!0===e?u(t).set(n,r):e[t._i]=r,t},ufstore:u}},function(t,n,r){"use strict";var p=r(123),d=r(5),y=r(8),g=r(47),b=r(7)("isConcatSpreadable");t.exports=function t(n,r,e,i,o,u,c,f){for(var a,s,l=o,h=0,v=!!c&&g(c,f,3);h<i;){if(h in e){if(a=v?v(e[h],h,r):e[h],s=!1,d(a)&&(s=void 0!==(s=a[b])?!!s:p(a)),s&&0<u)l=t(n,r,a,y(a.length),l,u-1)-1;else{if(9007199254740991<=l)throw TypeError();n[l]=a}l++}h++}return l}},function(t,n,r){t.exports=!r(10)&&!r(4)(function(){return 7!=Object.defineProperty(r(140)("div"),"a",{get:function(){return 7}}).a})},function(t,n){t.exports=function(t,n,r){var e=void 0===r;switch(n.length){case 0:return e?t():t.call(r);case 1:return e?t(n[0]):t.call(r,n[0]);case 2:return e?t(n[0],n[1]):t.call(r,n[0],n[1]);case 3:return e?t(n[0],n[1],n[2]):t.call(r,n[0],n[1],n[2]);case 4:return e?t(n[0],n[1],n[2],n[3]):t.call(r,n[0],n[1],n[2],n[3])}return t.apply(r,n)}},function(t,n,r){var e=r(5),i=Math.floor;t.exports=function(t){return!e(t)&&isFinite(t)&&i(t)===t}},function(t,n,r){var o=r(2);t.exports=function(t,n,r,e){try{return e?n(o(r)[0],r[1]):n(r)}catch(n){var i=t.return;throw void 0!==i&&o(i.call(t)),n}}},function(t,n){t.exports=function(t,n){return{value:n,done:!!t}}},function(t,n,r){var o=r(149),e=Math.pow,u=e(2,-52),c=e(2,-23),f=e(2,127)*(2-c),a=e(2,-126);t.exports=Math.fround||function(t){var n,r,e=Math.abs(t),i=o(t);return e<a?i*(e/a/c+1/u-1/u)*a*c:f<(r=(n=(1+c/u)*e)-(n-e))||r!=r?i*(1/0):i*r}},function(t,n){t.exports=Math.log1p||function(t){return-1e-8<(t=+t)&&t<1e-8?t-t*t/2:Math.log(1+t)}},function(t,n){t.exports=Math.scale||function(t,n,r,e,i){return 0===arguments.length||t!=t||n!=n||r!=r||e!=e||i!=i?NaN:t===1/0||t===-1/0?t:(t-n)*(i-e)/(r-n)+e}},function(t,n,r){"use strict";var h=r(10),v=r(74),p=r(127),d=r(117),y=r(17),g=r(116),i=Object.assign;t.exports=!i||r(4)(function(){var t={},n={},r=Symbol(),e="abcdefghijklmnopqrst";return t[r]=7,e.split("").forEach(function(t){n[t]=t}),7!=i({},t)[r]||Object.keys(i({},n)).join("")!=e})?function(t,n){for(var r=y(t),e=arguments.length,i=1,o=p.f,u=d.f;i<e;)for(var c,f=g(arguments[i++]),a=o?v(f).concat(o(f)):v(f),s=a.length,l=0;l<s;)c=a[l++],h&&!u.call(f,c)||(r[c]=f[c]);return r}:i},function(t,n,r){var u=r(11),c=r(2),f=r(74);t.exports=r(10)?Object.defineProperties:function(t,n){c(t);for(var r,e=f(n),i=e.length,o=0;o<i;)u.f(t,r=e[o++],n[r]);return t}},function(t,n,r){var e=r(33),i=r(73).f,o={}.toString,u="object"==typeof window&&window&&Object.getOwnPropertyNames?Object.getOwnPropertyNames(window):[];t.exports.f=function(t){return u&&"[object Window]"==o.call(t)?function(t){try{return i(t)}catch(t){return u.slice()}}(t):i(e(t))}},function(t,n,r){var u=r(30),c=r(33),f=r(120)(!1),a=r(154)("IE_PROTO");t.exports=function(t,n){var r,e=c(t),i=0,o=[];for(r in e)r!=a&&u(e,r)&&o.push(r);for(;n.length>i;)u(e,r=n[i++])&&(~f(o,r)||o.push(r));return o}},function(t,n,r){var f=r(10),a=r(74),s=r(33),l=r(117).f;t.exports=function(c){return function(t){for(var n,r=s(t),e=a(r),i=e.length,o=0,u=[];o<i;)n=e[o++],f&&!l.call(r,n)||u.push(c?[n,r[n]]:r[n]);return u}}},function(t,n,r){var e=r(73),i=r(127),o=r(2),u=r(3).Reflect;t.exports=u&&u.ownKeys||function(t){var n=e.f(o(t)),r=i.f;return r?n.concat(r(t)):n}},function(t,n,r){var e=r(3).parseFloat,i=r(84).trim;t.exports=1/e(r(157)+"-0")!=-1/0?function(t){var n=i(String(t),3),r=e(n);return 0===r&&"-"==n.charAt(0)?-0:r}:e},function(t,n,r){var e=r(3).parseInt,i=r(84).trim,o=r(157),u=/^[-+]?0[xX]/;t.exports=8!==e(o+"08")||22!==e(o+"0x16")?function(t,n){var r=i(String(t),3);return e(r,n>>>0||(u.test(r)?16:10))}:e},function(t,n){t.exports=function(t){try{return{e:!1,v:t()}}catch(t){return{e:!0,v:t}}}},function(t,n,r){var e=r(2),i=r(5),o=r(151);t.exports=function(t,n){if(e(t),i(n)&&n.constructor===t)return n;var r=o.f(t);return(0,r.resolve)(n),r.promise}},function(t,n){t.exports=Object.is||function(t,n){return t===n?0!==t||1/t==1/n:t!=t&&n!=n}},function(t,n,r){var s=r(8),l=r(156),h=r(51);t.exports=function(t,n,r,e){var i=String(h(t)),o=i.length,u=void 0===r?" ":String(r),c=s(n);if(c<=o||""==u)return i;var f=c-o,a=l.call(u,Math.ceil(f/u.length));return a.length>f&&(a=a.slice(0,f)),e?a+i:i+a}},function(t,n,r){var e=r(49),i=r(8);t.exports=function(t){if(void 0===t)return 0;var n=e(t),r=i(n);if(n!==r)throw RangeError("Wrong length!");return r}},function(t,n,r){n.f=r(7)},function(t,n,r){"use strict";var e=r(170),i=r(80);t.exports=r(121)("Map",function(t){return function(){return t(this,0<arguments.length?arguments[0]:void 0)}},{get:function(t){var n=e.getEntry(i(this,"Map"),t);return n&&n.v},set:function(t,n){return e.def(i(this,"Map"),0===t?0:t,n)}},e,!0)},function(t,n,r){"use strict";var e=r(152);r(1)({target:"RegExp",proto:!0,forced:e!==/./.exec},{exec:e})},function(t,n,r){r(10)&&"g"!=/./g.flags&&r(11).f(RegExp.prototype,"flags",{configurable:!0,get:r(115)})},function(t,n,r){"use strict";var e=r(170),i=r(80);t.exports=r(121)("Set",function(t){return function(){return t(this,0<arguments.length?arguments[0]:void 0)}},{add:function(t){return e.def(i(this,"Set"),t=0===t?0:t,t)}},e)},function(t,n,r){"use strict";function e(t){return function(){return t(this,0<arguments.length?arguments[0]:void 0)}}var o,i=r(3),u=r(50)(0),c=r(27),f=r(69),a=r(182),s=r(172),l=r(5),h=r(80),v=r(80),p=!i.ActiveXObject&&"ActiveXObject"in i,d="WeakMap",y=f.getWeak,g=Object.isExtensible,b=s.ufstore,x={get:function(t){if(l(t)){var n=y(t);return!0===n?b(h(this,d)).get(t):n?n[this._i]:void 0}},set:function(t,n){return s.def(h(this,d),t,n)}},m=t.exports=r(121)(d,e,x,s,!0,!0);v&&p&&(a((o=s.getConstructor(e,d)).prototype,x),f.NEED=!0,u(["delete","has","get","set"],function(e){var t=m.prototype,i=t[e];c(t,e,function(t,n){if(!l(t)||g(t))return i.call(this,t,n);this._f||(this._f=new o);var r=this._f[e](t,n);return"set"==e?this:r})}))},,,,function(t,n){"use strict";t.exports={init:function(){var t=document.querySelector("#page-nav");t&&!document.querySelector("#page-nav .extend.prev")&&(t.innerHTML='<a class="extend prev disabled" rel="prev">« Prev</a>'+t.innerHTML),t&&!document.querySelector("#page-nav .extend.next")&&(t.innerHTML=t.innerHTML+'<a class="extend next disabled" rel="next">Next »</a>'),yiliaConfig&&yiliaConfig.open_in_new&&document.querySelectorAll(".article-entry a:not(.article-more-a)").forEach(function(t){var n=t.getAttribute("target");n&&""!==n||t.setAttribute("target","_blank")}),yiliaConfig&&yiliaConfig.toc_hide_index&&document.querySelectorAll(".toc-number").forEach(function(t){t.style.display="none"})}}},function(t,n,r){"use strict";function e(t){return t&&t.__esModule?t:{default:t}}function i(r,t,n,e,i){var o,u,c=function(){for(var t=r.offsetLeft,n=r.offsetParent;null!==n;)t+=n.offsetLeft,n=n.offsetParent;return t}(),f=function(){for(var t=r.offsetTop,n=r.offsetParent;null!==n;)t+=n.offsetTop,n=n.offsetParent;return t}()-t;f-n<=i?((o=r.$newDom)||(o=r.cloneNode(!0),(0,a.default)(r,o),(r.$newDom=o).style.position="fixed",o.style.top=(n||f)+"px",o.style.left=c+"px",o.style.zIndex=e||2,o.style.width="100%",o.style.color="#fff"),o.style.visibility="visible",r.style.visibility="hidden"):(r.style.visibility="visible",(u=r.$newDom)&&(u.style.visibility="hidden"))}function o(){var t=document.querySelector(".js-overlay"),n=document.querySelector(".js-header-menu");i(t,document.body.scrollTop,-63,2,0),i(n,document.body.scrollTop,1,3,0)}var f=e(r(163)),a=e((e(r(164)),r(414))),u=e(r(134)),c=e(r(204)),s=r(135);u.default.versions.mobile&&window.screen.width<800&&(function(){for(var t=document.querySelectorAll(".js-header-menu li a"),n=window.location.pathname,r=0,e=t.length;r<e;r++){var i=t[r],o=n,u=i.getAttribute("href"),c=/\/|index.html/g;o.replace(c,"")===u.replace(c,"")&&(0,f.default)(i,"active")}}(),document.querySelector("#container").addEventListener("scroll",function(t){o()}),window.addEventListener("scroll",function(t){o()}),o()),(0,s.addLoadEvent)(function(){c.default.init()}),t.exports={}},,,function(t,n,r){(function(t){"use strict";function n(t,n,r){t[n]||Object.defineProperty(t,n,{writable:!0,configurable:!0,value:r})}if(r(413),r(209),r(211),t._babelPolyfill)throw new Error("only one instance of babel-polyfill is allowed");t._babelPolyfill=!0,n(String.prototype,"padLeft","".padStart),n(String.prototype,"padRight","".padEnd),"pop,reverse,shift,keys,values,entries,indexOf,every,some,forEach,map,filter,find,findIndex,includes,join,slice,concat,push,splice,unshift,sort,lastIndexOf,reduce,reduceRight,copyWithin,fill".split(",").forEach(function(t){[][t]&&n(Array,t,Function.call.bind([][t]))})}).call(n,function(){return this}())},function(I,t){(function(t){!function(t){"use strict";function o(t,n,r,e){var o,u,c,f,i=n&&n.prototype instanceof h?n:h,a=Object.create(i.prototype),s=new v(e||[]);return a._invoke=(o=t,u=r,c=s,f=y,function(t,n){if(f===b)throw new Error("Generator is already running");if(f===x){if("throw"===t)throw n;return p()}for(c.method=t,c.arg=n;;){var r=c.delegate;if(r){var e=function t(n,r){var e=n.iterator[r.method];if(e===d){if(r.delegate=null,"throw"===r.method){if(n.iterator.return&&(r.method="return",r.arg=d,t(n,r),"throw"===r.method))return m;r.method="throw",r.arg=new TypeError("The iterator does not provide a 'throw' method")}return m}var i=l(e,n.iterator,r.arg);if("throw"===i.type)return r.method="throw",r.arg=i.arg,r.delegate=null,m;var o=i.arg;return o?o.done?(r[n.resultName]=o.value,r.next=n.nextLoc,"return"!==r.method&&(r.method="next",r.arg=d),r.delegate=null,m):o:(r.method="throw",r.arg=new TypeError("iterator result is not an object"),r.delegate=null,m)}(r,c);if(e){if(e===m)continue;return e}}if("next"===c.method)c.sent=c._sent=c.arg;else if("throw"===c.method){if(f===y)throw f=x,c.arg;c.dispatchException(c.arg)}else"return"===c.method&&c.abrupt("return",c.arg);f=b;var i=l(o,u,c);if("normal"===i.type){if(f=c.done?x:g,i.arg===m)continue;return{value:i.arg,done:c.done}}"throw"===i.type&&(f=x,c.method="throw",c.arg=i.arg)}}),a}function l(t,n,r){try{return{type:"normal",arg:t.call(n,r)}}catch(t){return{type:"throw",arg:t}}}function h(){}function r(){}function n(){}function e(t){["next","throw","return"].forEach(function(n){t[n]=function(t){return this._invoke(n,t)}})}function u(c){function f(t,n,r,e){var i=l(c[t],c,n);if("throw"!==i.type){var o=i.arg,u=o.value;return u&&"object"==typeof u&&O.call(u,"__await")?Promise.resolve(u.__await).then(function(t){f("next",t,r,e)},function(t){f("throw",t,r,e)}):Promise.resolve(u).then(function(t){o.value=t,r(o)},e)}e(i.arg)}var n;"object"==typeof t.process&&t.process.domain&&(f=t.process.domain.bind(f)),this._invoke=function(r,e){function t(){return new Promise(function(t,n){f(r,e,t,n)})}return n=n?n.then(t,t):t()}}function i(t){var n={tryLoc:t[0]};1 in t&&(n.catchLoc=t[1]),2 in t&&(n.finallyLoc=t[2],n.afterLoc=t[3]),this.tryEntries.push(n)}function c(t){var n=t.completion||{};n.type="normal",delete n.arg,t.completion=n}function v(t){this.tryEntries=[{tryLoc:"root"}],t.forEach(i,this),this.reset(!0)}function f(n){if(n){var t=n[M];if(t)return t.call(n);if("function"==typeof n.next)return n;if(!isNaN(n.length)){var r=-1,e=function t(){for(;++r<n.length;)if(O.call(n,r))return t.value=n[r],t.done=!1,t;return t.value=d,t.done=!0,t};return e.next=e}}return{next:p}}function p(){return{value:d,done:!0}}var d,y,g,b,x,m,a,s,S,w,_=Object.prototype,O=_.hasOwnProperty,E="function"==typeof Symbol?Symbol:{},M=E.iterator||"@@iterator",P=E.asyncIterator||"@@asyncIterator",j=E.toStringTag||"@@toStringTag",F="object"==typeof I,A=t.regeneratorRuntime;A?F&&(I.exports=A):((A=t.regeneratorRuntime=F?I.exports:{}).wrap=o,y="suspendedStart",g="suspendedYield",b="executing",x="completed",m={},(a={})[M]=function(){return this},(S=(s=Object.getPrototypeOf)&&s(s(f([]))))&&S!==_&&O.call(S,M)&&(a=S),w=n.prototype=h.prototype=Object.create(a),(r.prototype=w.constructor=n).constructor=r,n[j]=r.displayName="GeneratorFunction",A.isGeneratorFunction=function(t){var n="function"==typeof t&&t.constructor;return!!n&&(n===r||"GeneratorFunction"===(n.displayName||n.name))},A.mark=function(t){return Object.setPrototypeOf?Object.setPrototypeOf(t,n):(t.__proto__=n,j in t||(t[j]="GeneratorFunction")),t.prototype=Object.create(w),t},A.awrap=function(t){return{__await:t}},e(u.prototype),u.prototype[P]=function(){return this},A.AsyncIterator=u,A.async=function(t,n,r,e){var i=new u(o(t,n,r,e));return A.isGeneratorFunction(n)?i:i.next().then(function(t){return t.done?t.value:i.next()})},e(w),w[j]="Generator",w[M]=function(){return this},w.toString=function(){return"[object Generator]"},A.keys=function(r){var e=[];for(var t in r)e.push(t);return e.reverse(),function t(){for(;e.length;){var n=e.pop();if(n in r)return t.value=n,t.done=!1,t}return t.done=!0,t}},A.values=f,v.prototype={constructor:v,reset:function(t){if(this.prev=0,this.next=0,this.sent=this._sent=d,this.done=!1,this.delegate=null,this.method="next",this.arg=d,this.tryEntries.forEach(c),!t)for(var n in this)"t"===n.charAt(0)&&O.call(this,n)&&!isNaN(+n.slice(1))&&(this[n]=d)},stop:function(){this.done=!0;var t=this.tryEntries[0].completion;if("throw"===t.type)throw t.arg;return this.rval},dispatchException:function(r){function t(t,n){return o.type="throw",o.arg=r,e.next=t,n&&(e.method="next",e.arg=d),!!n}if(this.done)throw r;for(var e=this,n=this.tryEntries.length-1;0<=n;--n){var i=this.tryEntries[n],o=i.completion;if("root"===i.tryLoc)return t("end");if(i.tryLoc<=this.prev){var u=O.call(i,"catchLoc"),c=O.call(i,"finallyLoc");if(u&&c){if(this.prev<i.catchLoc)return t(i.catchLoc,!0);if(this.prev<i.finallyLoc)return t(i.finallyLoc)}else if(u){if(this.prev<i.catchLoc)return t(i.catchLoc,!0)}else{if(!c)throw new Error("try statement without catch or finally");if(this.prev<i.finallyLoc)return t(i.finallyLoc)}}}},abrupt:function(t,n){for(var r=this.tryEntries.length-1;0<=r;--r){var e=this.tryEntries[r];if(e.tryLoc<=this.prev&&O.call(e,"finallyLoc")&&this.prev<e.finallyLoc){var i=e;break}}i&&("break"===t||"continue"===t)&&i.tryLoc<=n&&n<=i.finallyLoc&&(i=null);var o=i?i.completion:{};return o.type=t,o.arg=n,i?(this.method="next",this.next=i.finallyLoc,m):this.complete(o)},complete:function(t,n){if("throw"===t.type)throw t.arg;return"break"===t.type||"continue"===t.type?this.next=t.arg:"return"===t.type?(this.rval=this.arg=t.arg,this.method="return",this.next="end"):"normal"===t.type&&n&&(this.next=n),m},finish:function(t){for(var n=this.tryEntries.length-1;0<=n;--n){var r=this.tryEntries[n];if(r.finallyLoc===t)return this.complete(r.completion,r.afterLoc),c(r),m}},catch:function(t){for(var n=this.tryEntries.length-1;0<=n;--n){var r=this.tryEntries[n];if(r.tryLoc===t){var e,i=r.completion;return"throw"===i.type&&(e=i.arg,c(r)),e}}throw new Error("illegal catch attempt")},delegateYield:function(t,n,r){return this.delegate={iterator:f(t),resultName:n,nextLoc:r},"next"===this.method&&(this.arg=d),m}})}("object"==typeof t?t:"object"==typeof window?window:"object"==typeof self?self:this)}).call(t,function(){return this}())},,function(t,n,r){r(221),t.exports=r(46).RegExp.escape},,,,function(t,n,r){var e=r(5),i=r(123),o=r(7)("species");t.exports=function(t){var n;return i(t)&&("function"!=typeof(n=t.constructor)||n!==Array&&!i(n.prototype)||(n=void 0),e(n)&&null===(n=n[o])&&(n=void 0)),void 0===n?Array:n}},function(t,n,r){"use strict";function i(t){return 9<t?t:"0"+t}var e=r(4),o=Date.prototype.getTime,u=Date.prototype.toISOString;t.exports=e(function(){return"0385-07-25T07:06:39.999Z"!=u.call(new Date(-5e13-1))})||!e(function(){u.call(new Date(NaN))})?function(){if(!isFinite(o.call(this)))throw RangeError("Invalid time value");var t=this,n=t.getUTCFullYear(),r=t.getUTCMilliseconds(),e=n<0?"-":9999<n?"+":"";return e+("00000"+Math.abs(n)).slice(e?-6:-4)+"-"+i(t.getUTCMonth()+1)+"-"+i(t.getUTCDate())+"T"+i(t.getUTCHours())+":"+i(t.getUTCMinutes())+":"+i(t.getUTCSeconds())+"."+(99<r?r:"0"+i(r))+"Z"}:u},function(t,n,r){"use strict";var e=r(2),i=r(53);t.exports=function(t){if("string"!==t&&"number"!==t&&"default"!==t)throw TypeError("Incorrect hint");return i(e(this),"number"!=t)}},function(t,n,r){var c=r(74),f=r(127),a=r(117);t.exports=function(t){var n=c(t),r=f.f;if(r)for(var e,i=r(t),o=a.f,u=0;i.length>u;)o.call(t,e=i[u++])&&n.push(e);return n}},function(t,n,r){t.exports=r(118)("native-function-to-string",Function.toString)},function(t,n){t.exports=function(n,r){var e=r===Object(r)?function(t){return r[t]}:r;return function(t){return String(t).replace(n,e)}}},function(t,n,r){var e=r(1),i=r(220)(/[\\^$*+?.()|[\]{}]/g,"\\$&");e(e.S,"RegExp",{escape:function(t){return i(t)}})},function(t,n,r){var e=r(1);e(e.P,"Array",{copyWithin:r(166)}),r(67)("copyWithin")},function(t,n,r){"use strict";var e=r(1),i=r(50)(4);e(e.P+e.F*!r(48)([].every,!0),"Array",{every:function(t){return i(this,t,arguments[1])}})},function(t,n,r){var e=r(1);e(e.P,"Array",{fill:r(137)}),r(67)("fill")},function(t,n,r){"use strict";var e=r(1),i=r(50)(2);e(e.P+e.F*!r(48)([].filter,!0),"Array",{filter:function(t){return i(this,t,arguments[1])}})},function(t,n,r){"use strict";var e=r(1),i=r(50)(6),o="findIndex",u=!0;o in[]&&Array(1)[o](function(){u=!1}),e(e.P+e.F*u,"Array",{findIndex:function(t){return i(this,t,1<arguments.length?arguments[1]:void 0)}}),r(67)(o)},function(t,n,r){"use strict";var e=r(1),i=r(50)(5),o="find",u=!0;o in[]&&Array(1)[o](function(){u=!1}),e(e.P+e.F*u,"Array",{find:function(t){return i(this,t,1<arguments.length?arguments[1]:void 0)}}),r(67)(o)},function(t,n,r){"use strict";var e=r(1),i=r(50)(0),o=r(48)([].forEach,!0);e(e.P+e.F*!o,"Array",{forEach:function(t){return i(this,t,arguments[1])}})},function(t,n,r){"use strict";var h=r(47),e=r(1),v=r(17),p=r(177),d=r(145),y=r(8),g=r(139),b=r(161);e(e.S+e.F*!r(125)(function(t){Array.from(t)}),"Array",{from:function(t){var n,r,e,i,o=v(t),u="function"==typeof this?this:Array,c=arguments.length,f=1<c?arguments[1]:void 0,a=void 0!==f,s=0,l=b(o);if(a&&(f=h(f,2<c?arguments[2]:void 0,2)),null==l||u==Array&&d(l))for(r=new u(n=y(o.length));s<n;s++)g(r,s,a?f(o[s],s):o[s]);else for(i=l.call(o),r=new u;!(e=i.next()).done;s++)g(r,s,a?p(i,f,[e.value,s],!0):e.value);return r.length=s,r}})},function(t,n,r){"use strict";var e=r(1),i=r(120)(!1),o=[].indexOf,u=!!o&&1/[1].indexOf(1,-0)<0;e(e.P+e.F*(u||!r(48)(o)),"Array",{indexOf:function(t){return u?o.apply(this,arguments)||0:i(this,t,arguments[1])}})},function(t,n,r){var e=r(1);e(e.S,"Array",{isArray:r(123)})},function(t,n,r){"use strict";var e=r(1),i=r(33),o=[].join;e(e.P+e.F*(r(116)!=Object||!r(48)(o)),"Array",{join:function(t){return o.call(i(this),void 0===t?",":t)}})},function(t,n,r){"use strict";var e=r(1),i=r(33),o=r(49),u=r(8),c=[].lastIndexOf,f=!!c&&1/[1].lastIndexOf(1,-0)<0;e(e.P+e.F*(f||!r(48)(c)),"Array",{lastIndexOf:function(t){if(f)return c.apply(this,arguments)||0;var n=i(this),r=u(n.length),e=r-1;for(1<arguments.length&&(e=Math.min(e,o(arguments[1]))),e<0&&(e=r+e);0<=e;e--)if(e in n&&n[e]===t)return e||0;return-1}})},function(t,n,r){"use strict";var e=r(1),i=r(50)(1);e(e.P+e.F*!r(48)([].map,!0),"Array",{map:function(t){return i(this,t,arguments[1])}})},function(t,n,r){"use strict";var e=r(1),i=r(139);e(e.S+e.F*r(4)(function(){function t(){}return!(Array.of.call(t)instanceof t)}),"Array",{of:function(){for(var t=0,n=arguments.length,r=new("function"==typeof this?this:Array)(n);t<n;)i(r,t,arguments[t++]);return r.length=n,r}})},function(t,n,r){"use strict";var e=r(1),i=r(168);e(e.P+e.F*!r(48)([].reduceRight,!0),"Array",{reduceRight:function(t){return i(this,t,arguments.length,arguments[1],!0)}})},function(t,n,r){"use strict";var e=r(1),i=r(168);e(e.P+e.F*!r(48)([].reduce,!0),"Array",{reduce:function(t){return i(this,t,arguments.length,arguments[1],!1)}})},function(t,n,r){"use strict";var e=r(1),i=r(143),a=r(45),s=r(78),l=r(8),h=[].slice;e(e.P+e.F*r(4)(function(){i&&h.call(i)}),"Array",{slice:function(t,n){var r=l(this.length),e=a(this);if(n=void 0===n?r:n,"Array"==e)return h.call(this,t,n);for(var i=s(t,r),o=s(n,r),u=l(o-i),c=new Array(u),f=0;f<u;f++)c[f]="String"==e?this.charAt(i+f):this[i+f];return c}})},function(t,n,r){"use strict";var e=r(1),i=r(50)(3);e(e.P+e.F*!r(48)([].some,!0),"Array",{some:function(t){return i(this,t,arguments[1])}})},function(t,n,r){"use strict";var e=r(1),i=r(21),o=r(17),u=r(4),c=[].sort,f=[1,2,3];e(e.P+e.F*(u(function(){f.sort(void 0)})||!u(function(){f.sort(null)})||!r(48)(c)),"Array",{sort:function(t){return void 0===t?c.call(o(this)):c.call(o(this),i(t))}})},function(t,n,r){r(77)("Array")},function(t,n,r){var e=r(1);e(e.S,"Date",{now:function(){return(new Date).getTime()}})},function(t,n,r){var e=r(1),i=r(216);e(e.P+e.F*(Date.prototype.toISOString!==i),"Date",{toISOString:i})},function(t,n,r){"use strict";var e=r(1),i=r(17),o=r(53);e(e.P+e.F*r(4)(function(){return null!==new Date(NaN).toJSON()||1!==Date.prototype.toJSON.call({toISOString:function(){return 1}})}),"Date",{toJSON:function(t){var n=i(this),r=o(n);return"number"!=typeof r||isFinite(r)?n.toISOString():null}})},function(t,n,r){var e=r(7)("toPrimitive"),i=Date.prototype;e in i||r(26)(i,e,r(217))},function(t,n,r){var e=Date.prototype,i="Invalid Date",o="toString",u=e[o],c=e.getTime;new Date(NaN)+""!=i&&r(27)(e,o,function(){var t=c.call(this);return t==t?u.call(this):i})},function(t,n,r){var e=r(1);e(e.P,"Function",{bind:r(169)})},function(t,n,r){"use strict";var e=r(5),i=r(32),o=r(7)("hasInstance"),u=Function.prototype;o in u||r(11).f(u,o,{value:function(t){if("function"!=typeof this||!e(t))return!1;if(!e(this.prototype))return t instanceof this;for(;t=i(t);)if(this.prototype===t)return!0;return!1}})},function(t,n,r){var e=r(11).f,i=Function.prototype,o=/^\s*function ([^ (]*)/;"name"in i||r(10)&&e(i,"name",{configurable:!0,get:function(){try{return(""+this).match(o)[1]}catch(t){return""}}})},function(t,n,r){var e=r(1),i=r(180),o=Math.sqrt,u=Math.acosh;e(e.S+e.F*!(u&&710==Math.floor(u(Number.MAX_VALUE))&&u(1/0)==1/0),"Math",{acosh:function(t){return(t=+t)<1?NaN:94906265.62425156<t?Math.log(t)+Math.LN2:i(t-1+o(t-1)*o(t+1))}})},function(t,n,r){var e=r(1),i=Math.asinh;e(e.S+e.F*!(i&&0<1/i(0)),"Math",{asinh:function t(n){return isFinite(n=+n)&&0!=n?n<0?-t(-n):Math.log(n+Math.sqrt(n*n+1)):n}})},function(t,n,r){var e=r(1),i=Math.atanh;e(e.S+e.F*!(i&&1/i(-0)<0),"Math",{atanh:function(t){return 0==(t=+t)?t:Math.log((1+t)/(1-t))/2}})},function(t,n,r){var e=r(1),i=r(149);e(e.S,"Math",{cbrt:function(t){return i(t=+t)*Math.pow(Math.abs(t),1/3)}})},function(t,n,r){var e=r(1);e(e.S,"Math",{clz32:function(t){return(t>>>=0)?31-Math.floor(Math.log(t+.5)*Math.LOG2E):32}})},function(t,n,r){var e=r(1),i=Math.exp;e(e.S,"Math",{cosh:function(t){return(i(t=+t)+i(-t))/2}})},function(t,n,r){var e=r(1),i=r(148);e(e.S+e.F*(i!=Math.expm1),"Math",{expm1:i})},function(t,n,r){var e=r(1);e(e.S,"Math",{fround:r(179)})},function(t,n,r){var e=r(1),f=Math.abs;e(e.S,"Math",{hypot:function(t,n){for(var r,e,i=0,o=0,u=arguments.length,c=0;o<u;)c<(r=f(arguments[o++]))?(i=i*(e=c/r)*e+1,c=r):i+=0<r?(e=r/c)*e:r;return c===1/0?1/0:c*Math.sqrt(i)}})},function(t,n,r){var e=r(1),i=Math.imul;e(e.S+e.F*r(4)(function(){return-5!=i(4294967295,5)||2!=i.length}),"Math",{imul:function(t,n){var r=65535,e=+t,i=+n,o=r&e,u=r&i;return 0|o*u+((r&e>>>16)*u+o*(r&i>>>16)<<16>>>0)}})},function(t,n,r){var e=r(1);e(e.S,"Math",{log10:function(t){return Math.log(t)*Math.LOG10E}})},function(t,n,r){var e=r(1);e(e.S,"Math",{log1p:r(180)})},function(t,n,r){var e=r(1);e(e.S,"Math",{log2:function(t){return Math.log(t)/Math.LN2}})},function(t,n,r){var e=r(1);e(e.S,"Math",{sign:r(149)})},function(t,n,r){var e=r(1),i=r(148),o=Math.exp;e(e.S+e.F*r(4)(function(){return-2e-17!=!Math.sinh(-2e-17)}),"Math",{sinh:function(t){return Math.abs(t=+t)<1?(i(t)-i(-t))/2:(o(t-1)-o(-t-1))*(Math.E/2)}})},function(t,n,r){var e=r(1),i=r(148),o=Math.exp;e(e.S,"Math",{tanh:function(t){var n=i(t=+t),r=i(-t);return n==1/0?1:r==1/0?-1:(n-r)/(o(t)+o(-t))}})},function(t,n,r){var e=r(1);e(e.S,"Math",{trunc:function(t){return(0<t?Math.floor:Math.ceil)(t)}})},function(t,n,r){"use strict";function e(t){var n=s(t,!1);if("string"==typeof n&&2<n.length){var r,e,i,o=(n=x?n.trim():v(n,3)).charCodeAt(0);if(43===o||45===o){if(88===(r=n.charCodeAt(2))||120===r)return NaN}else if(48===o){switch(n.charCodeAt(1)){case 66:case 98:e=2,i=49;break;case 79:case 111:e=8,i=55;break;default:return+n}for(var u,c=n.slice(2),f=0,a=c.length;f<a;f++)if((u=c.charCodeAt(f))<48||i<u)return NaN;return parseInt(c,e)}}return+n}var i=r(3),o=r(30),u=r(45),c=r(144),s=r(53),f=r(4),a=r(73).f,l=r(31).f,h=r(11).f,v=r(84).trim,p="Number",d=i[p],y=d,g=d.prototype,b=u(r(72)(g))==p,x="trim"in String.prototype;if(!d(" 0o1")||!d("0b1")||d("+0x1")){d=function(t){var n=arguments.length<1?0:t,r=this;return r instanceof d&&(b?f(function(){g.valueOf.call(r)}):u(r)!=p)?c(new y(e(n)),r,d):e(n)};for(var m,S=r(10)?a(y):"MAX_VALUE,MIN_VALUE,NaN,NEGATIVE_INFINITY,POSITIVE_INFINITY,EPSILON,isFinite,isInteger,isNaN,isSafeInteger,MAX_SAFE_INTEGER,MIN_SAFE_INTEGER,parseFloat,parseInt,isInteger".split(","),w=0;S.length>w;w++)o(y,m=S[w])&&!o(d,m)&&h(d,m,l(y,m));(d.prototype=g).constructor=d,r(27)(i,p,d)}},function(t,n,r){var e=r(1);e(e.S,"Number",{EPSILON:Math.pow(2,-52)})},function(t,n,r){var e=r(1),i=r(3).isFinite;e(e.S,"Number",{isFinite:function(t){return"number"==typeof t&&i(t)}})},function(t,n,r){var e=r(1);e(e.S,"Number",{isInteger:r(176)})},function(t,n,r){var e=r(1);e(e.S,"Number",{isNaN:function(t){return t!=t}})},function(t,n,r){var e=r(1),i=r(176),o=Math.abs;e(e.S,"Number",{isSafeInteger:function(t){return i(t)&&o(t)<=9007199254740991}})},function(t,n,r){var e=r(1);e(e.S,"Number",{MAX_SAFE_INTEGER:9007199254740991})},function(t,n,r){var e=r(1);e(e.S,"Number",{MIN_SAFE_INTEGER:-9007199254740991})},function(t,n,r){var e=r(1),i=r(188);e(e.S+e.F*(Number.parseFloat!=i),"Number",{parseFloat:i})},function(t,n,r){var e=r(1),i=r(189);e(e.S+e.F*(Number.parseInt!=i),"Number",{parseInt:i})},function(t,n,r){"use strict";function a(t,n){for(var r=-1,e=n;++r<6;)e+=t*u[r],u[r]=e%1e7,e=o(e/1e7)}function s(t){for(var n=6,r=0;0<=--n;)r+=u[n],u[n]=o(r/t),r=r%t*1e7}function l(){for(var t,n=6,r="";0<=--n;)""===r&&0!==n&&0===u[n]||(t=String(u[n]),r=""===r?t:r+p.call("0",7-t.length)+t);return r}var e=r(1),h=r(49),v=r(165),p=r(156),i=1..toFixed,o=Math.floor,u=[0,0,0,0,0,0],d="Number.toFixed: incorrect invocation!",y=function(t,n,r){return 0===n?r:n%2==1?y(t,n-1,r*t):y(t*t,n/2,r)};e(e.P+e.F*(!!i&&("0.000"!==8e-5.toFixed(3)||"1"!==.9.toFixed(0)||"1.25"!==1.255.toFixed(2)||"1000000000000000128"!==(0xde0b6b3a7640080).toFixed(0))||!r(4)(function(){i.call({})})),"Number",{toFixed:function(t){var n,r,e,i,o=v(this,d),u=h(t),c="",f="0";if(u<0||20<u)throw RangeError(d);if(o!=o)return"NaN";if(o<=-1e21||1e21<=o)return String(o);if(o<0&&(c="-",o=-o),1e-21<o)if(r=(n=function(){for(var t=0,n=o*y(2,69,1);4096<=n;)t+=12,n/=4096;for(;2<=n;)t+=1,n/=2;return t}()-69)<0?o*y(2,-n,1):o/y(2,n,1),r*=4503599627370496,0<(n=52-n)){for(a(0,r),e=u;7<=e;)a(1e7,0),e-=7;for(a(y(10,e,1),0),e=n-1;23<=e;)s(1<<23),e-=23;s(1<<e),a(1,1),s(2),f=l()}else a(0,r),a(1<<-n,0),f=l()+p.call("0",u);return 0<u?c+((i=f.length)<=u?"0."+p.call("0",u-i)+f:f.slice(0,i-u)+"."+f.slice(i-u)):c+f}})},function(t,n,r){"use strict";var e=r(1),i=r(4),o=r(165),u=1..toPrecision;e(e.P+e.F*(i(function(){return"1"!==u.call(1,void 0)})||!i(function(){u.call({})})),"Number",{toPrecision:function(t){var n=o(this,"Number#toPrecision: incorrect invocation!");return void 0===t?u.call(n):u.call(n,t)}})},function(t,n,r){var e=r(1);e(e.S+e.F,"Object",{assign:r(182)})},function(t,n,r){var e=r(1);e(e.S,"Object",{create:r(72)})},function(t,n,r){var e=r(1);e(e.S+e.F*!r(10),"Object",{defineProperties:r(183)})},function(t,n,r){var e=r(1);e(e.S+e.F*!r(10),"Object",{defineProperty:r(11).f})},function(t,n,r){var e=r(5),i=r(69).onFreeze;r(52)("freeze",function(n){return function(t){return n&&e(t)?n(i(t)):t}})},function(t,n,r){var e=r(33),i=r(31).f;r(52)("getOwnPropertyDescriptor",function(){return function(t,n){return i(e(t),n)}})},function(t,n,r){r(52)("getOwnPropertyNames",function(){return r(184).f})},function(t,n,r){var e=r(17),i=r(32);r(52)("getPrototypeOf",function(){return function(t){return i(e(t))}})},function(t,n,r){var e=r(5);r(52)("isExtensible",function(n){return function(t){return!!e(t)&&(!n||n(t))}})},function(t,n,r){var e=r(5);r(52)("isFrozen",function(n){return function(t){return!e(t)||!!n&&n(t)}})},function(t,n,r){var e=r(5);r(52)("isSealed",function(n){return function(t){return!e(t)||!!n&&n(t)}})},function(t,n,r){var e=r(1);e(e.S,"Object",{is:r(192)})},function(t,n,r){var e=r(17),i=r(74);r(52)("keys",function(){return function(t){return i(e(t))}})},function(t,n,r){var e=r(5),i=r(69).onFreeze;r(52)("preventExtensions",function(n){return function(t){return n&&e(t)?n(i(t)):t}})},function(t,n,r){var e=r(5),i=r(69).onFreeze;r(52)("seal",function(n){return function(t){return n&&e(t)?n(i(t)):t}})},function(t,n,r){var e=r(1);e(e.S,"Object",{setPrototypeOf:r(153).set})},function(t,n,r){"use strict";var e=r(81),i={};i[r(7)("toStringTag")]="z",i+""!="[object z]"&&r(27)(Object.prototype,"toString",function(){return"[object "+e(this)+"]"},!0)},function(t,n,r){var e=r(1),i=r(188);e(e.G+e.F*(parseFloat!=i),{parseFloat:i})},function(t,n,r){var e=r(1),i=r(189);e(e.G+e.F*(parseInt!=i),{parseInt:i})},function(t,n,r){"use strict";function e(){}function l(t){var n;return!(!y(t)||"function"!=typeof(n=t.then))&&n}function i(s,n){var r;s._n||(s._n=!0,r=s._c,w(function(){for(var f=s._v,a=1==s._s,t=0;r.length>t;)!function(t){var n,r,e,i=a?t.ok:t.fail,o=t.resolve,u=t.reject,c=t.domain;try{i?(a||(2==s._h&&D(s),s._h=1),!0===i?n=f:(c&&c.enter(),n=i(f),c&&(c.exit(),e=!0)),n===t.promise?u(j("Promise-chain cycle")):(r=l(n))?r.call(n,o,u):o(n)):u(f)}catch(t){c&&!e&&c.exit(),u(t)}}(r[t++]);s._c=[],s._n=!1,n&&!s._h&&R(s)}))}function o(t){var n=this;n._d||(n._d=!0,(n=n._w||n)._v=t,n._s=2,n._a||(n._a=n._c.slice()),i(n,!0))}var u,c,f,a,s=r(68),h=r(3),v=r(47),p=r(81),d=r(1),y=r(5),g=r(21),b=r(70),x=r(71),m=r(119),S=r(158).set,w=r(150)(),_=r(151),O=r(190),E=r(133),M=r(191),P="Promise",j=h.TypeError,F=h.process,A=F&&F.versions,I=A&&A.v8||"",L=h[P],N="process"==p(F),T=c=_.f,k=!!function(){try{var t=L.resolve(1),n=(t.constructor={})[r(7)("species")]=function(t){t(e,e)};return(N||"function"==typeof PromiseRejectionEvent)&&t.then(e)instanceof n&&0!==I.indexOf("6.6")&&-1===E.indexOf("Chrome/66")}catch(t){}}(),R=function(o){S.call(h,function(){var t,n,r,e=o._v,i=C(o);if(i&&(t=O(function(){N?F.emit("unhandledRejection",e,o):(n=h.onunhandledrejection)?n({promise:o,reason:e}):(r=h.console)&&r.error&&r.error("Unhandled promise rejection",e)}),o._h=N||C(o)?2:1),o._a=void 0,i&&t.e)throw t.v})},C=function(t){return 1!==t._h&&0===(t._a||t._c).length},D=function(n){S.call(h,function(){var t;N?F.emit("rejectionHandled",n):(t=h.onrejectionhandled)&&t({promise:n,reason:n._v})})},G=function(t){var r,e=this;if(!e._d){e._d=!0,e=e._w||e;try{if(e===t)throw j("Promise can't be resolved itself");(r=l(t))?w(function(){var n={_w:e,_d:!1};try{r.call(t,v(G,n,1),v(o,n,1))}catch(t){o.call(n,t)}}):(e._v=t,e._s=1,i(e,!1))}catch(t){o.call({_w:e,_d:!1},t)}}};k||(L=function(t){b(this,L,P,"_h"),g(t),u.call(this);try{t(v(G,this,1),v(o,this,1))}catch(t){o.call(this,t)}},(u=function(t){this._c=[],this._a=void 0,this._s=0,this._d=!1,this._v=void 0,this._h=0,this._n=!1}).prototype=r(76)(L.prototype,{then:function(t,n){var r=T(m(this,L));return r.ok="function"!=typeof t||t,r.fail="function"==typeof n&&n,r.domain=N?F.domain:void 0,this._c.push(r),this._a&&this._a.push(r),this._s&&i(this,!1),r.promise},catch:function(t){return this.then(void 0,t)}}),f=function(){var t=new u;this.promise=t,this.resolve=v(G,t,1),this.reject=v(o,t,1)},_.f=T=function(t){return t===L||t===a?new f:c(t)}),d(d.G+d.W+d.F*!k,{Promise:L}),r(83)(L,P),r(77)(P),a=r(46)[P],d(d.S+d.F*!k,P,{reject:function(t){var n=T(this);return(0,n.reject)(t),n.promise}}),d(d.S+d.F*(s||!k),P,{resolve:function(t){return M(s&&this===a?L:this,t)}}),d(d.S+d.F*!(k&&r(125)(function(t){L.all(t).catch(e)})),P,{all:function(t){var u=this,n=T(u),c=n.resolve,f=n.reject,r=O(function(){var e=[],i=0,o=1;x(t,!1,function(t){var n=i++,r=!1;e.push(void 0),o++,u.resolve(t).then(function(t){r||(r=!0,e[n]=t,--o||c(e))},f)}),--o||c(e)});return r.e&&f(r.v),n.promise},race:function(t){var n=this,r=T(n),e=r.reject,i=O(function(){x(t,!1,function(t){n.resolve(t).then(r.resolve,e)})});return i.e&&e(i.v),r.promise}})},function(t,n,r){var e=r(1),o=r(21),u=r(2),c=(r(3).Reflect||{}).apply,f=Function.apply;e(e.S+e.F*!r(4)(function(){c(function(){})}),"Reflect",{apply:function(t,n,r){var e=o(t),i=u(r);return c?c(e,n,i):f.call(e,n,i)}})},function(t,n,r){var e=r(1),c=r(72),f=r(21),a=r(2),s=r(5),i=r(4),l=r(169),h=(r(3).Reflect||{}).construct,v=i(function(){function t(){}return!(h(function(){},[],t)instanceof t)}),p=!i(function(){h(function(){})});e(e.S+e.F*(v||p),"Reflect",{construct:function(t,n){f(t),a(n);var r=arguments.length<3?t:f(arguments[2]);if(p&&!v)return h(t,n,r);if(t==r){switch(n.length){case 0:return new t;case 1:return new t(n[0]);case 2:return new t(n[0],n[1]);case 3:return new t(n[0],n[1],n[2]);case 4:return new t(n[0],n[1],n[2],n[3])}var e=[null];return e.push.apply(e,n),new(l.apply(t,e))}var i=r.prototype,o=c(s(i)?i:Object.prototype),u=Function.apply.call(t,o,n);return s(u)?u:o}})},function(t,n,r){var e=r(11),i=r(1),o=r(2),u=r(53);i(i.S+i.F*r(4)(function(){Reflect.defineProperty(e.f({},1,{value:1}),1,{value:2})}),"Reflect",{defineProperty:function(t,n,r){o(t),n=u(n,!0),o(r);try{return e.f(t,n,r),!0}catch(t){return!1}}})},function(t,n,r){var e=r(1),i=r(31).f,o=r(2);e(e.S,"Reflect",{deleteProperty:function(t,n){var r=i(o(t),n);return!(r&&!r.configurable)&&delete t[n]}})},function(t,n,r){"use strict";function e(t){this._t=o(t),this._i=0;var n,r=this._k=[];for(n in t)r.push(n)}var i=r(1),o=r(2);r(146)(e,"Object",function(){var t,n=this._k;do{if(this._i>=n.length)return{value:void 0,done:!0}}while(!((t=n[this._i++])in this._t));return{value:t,done:!1}}),i(i.S,"Reflect",{enumerate:function(t){return new e(t)}})},function(t,n,r){var e=r(31),i=r(1),o=r(2);i(i.S,"Reflect",{getOwnPropertyDescriptor:function(t,n){return e.f(o(t),n)}})},function(t,n,r){var e=r(1),i=r(32),o=r(2);e(e.S,"Reflect",{getPrototypeOf:function(t){return i(o(t))}})},function(t,n,r){var c=r(31),f=r(32),a=r(30),e=r(1),s=r(5),l=r(2);e(e.S,"Reflect",{get:function t(n,r,e){var i,o,u=arguments.length<3?n:e;return l(n)===u?n[r]:(i=c.f(n,r))?a(i,"value")?i.value:void 0!==i.get?i.get.call(u):void 0:s(o=f(n))?t(o,r,u):void 0}})},function(t,n,r){var e=r(1);e(e.S,"Reflect",{has:function(t,n){return n in t}})},function(t,n,r){var e=r(1),i=r(2),o=Object.isExtensible;e(e.S,"Reflect",{isExtensible:function(t){return i(t),!o||o(t)}})},function(t,n,r){var e=r(1);e(e.S,"Reflect",{ownKeys:r(187)})},function(t,n,r){var e=r(1),i=r(2),o=Object.preventExtensions;e(e.S,"Reflect",{preventExtensions:function(t){i(t);try{return o&&o(t),!0}catch(t){return!1}}})},function(t,n,r){var e=r(1),i=r(153);i&&e(e.S,"Reflect",{setPrototypeOf:function(t,n){i.check(t,n);try{return i.set(t,n),!0}catch(t){return!1}}})},function(t,n,r){var a=r(11),s=r(31),l=r(32),h=r(30),e=r(1),v=r(75),p=r(2),d=r(5);e(e.S,"Reflect",{set:function t(n,r,e,i){var o,u,c=arguments.length<4?n:i,f=s.f(p(n),r);if(!f){if(d(u=l(n)))return t(u,r,e,c);f=v(0)}if(h(f,"value")){if(!1===f.writable||!d(c))return!1;if(o=s.f(c,r)){if(o.get||o.set||!1===o.writable)return!1;o.value=e,a.f(c,r,o)}else a.f(c,r,v(0,e));return!0}return void 0!==f.set&&(f.set.call(c,e),!0)}})},function(t,n,r){var e=r(3),o=r(144),i=r(11).f,u=r(73).f,c=r(124),f=r(115),a=e.RegExp,s=a,l=a.prototype,h=/a/g,v=/a/g,p=new a(h)!==h;if(r(10)&&(!p||r(4)(function(){return v[r(7)("match")]=!1,a(h)!=h||a(v)==v||"/a/i"!=a(h,"i")}))){a=function(t,n){var r=this instanceof a,e=c(t),i=void 0===n;return!r&&e&&t.constructor===a&&i?t:o(p?new s(e&&!i?t.source:t,n):s((e=t instanceof a)?t.source:t,e&&i?f.call(t):n),r?this:l,a)};for(var d=u(s),y=0;d.length>y;)!function(n){n in a||i(a,n,{configurable:!0,get:function(){return s[n]},set:function(t){s[n]=t}})}(d[y++]);(l.constructor=a).prototype=l,r(27)(e,"RegExp",a)}r(77)("RegExp")},function(t,n,r){"use strict";var l=r(2),h=r(8),v=r(136),p=r(128);r(122)("match",1,function(e,i,a,s){return[function(t){var n=e(this),r=null==t?void 0:t[i];return void 0!==r?r.call(t,n):new RegExp(t)[i](String(n))},function(t){var n=s(a,t,this);if(n.done)return n.value;var r=l(t),e=String(this);if(!r.global)return p(r,e);for(var i,o=r.unicode,u=[],c=r.lastIndex=0;null!==(i=p(r,e));){var f=String(i[0]);""===(u[c]=f)&&(r.lastIndex=v(e,h(r.lastIndex),o)),c++}return 0===c?null:u}]})},function(t,n,r){"use strict";var _=r(2),O=r(17),E=r(8),M=r(49),P=r(136),j=r(128),F=Math.max,A=Math.min,I=Math.floor,L=/\$([$&`']|\d\d?|<[^>]*>)/g,N=/\$([$&`']|\d\d?)/g;r(122)("replace",2,function(i,o,S,w){return[function(t,n){var r=i(this),e=null==t?void 0:t[o];return void 0!==e?e.call(t,r,n):S.call(String(r),t,n)},function(t,n){var r=w(S,t,this,n);if(r.done)return r.value;var e=_(t),i=String(this),o="function"==typeof n;o||(n=String(n));var u,c,f=e.global;f&&(c=e.unicode,e.lastIndex=0);for(var a=[];;){var s=j(e,i);if(null===s)break;if(a.push(s),!f)break;""===String(s[0])&&(e.lastIndex=P(i,E(e.lastIndex),c))}for(var l="",h=0,v=0;v<a.length;v++){s=a[v];for(var p=String(s[0]),d=F(A(M(s.index),i.length),0),y=[],g=1;g<s.length;g++)y.push(void 0===(u=s[g])?u:String(u));var b,x=s.groups,m=o?(b=[p].concat(y,d,i),void 0!==x&&b.push(x),String(n.apply(void 0,b))):function(o,u,c,f,a,t){var s=c+o.length,l=f.length,n=N;return void 0!==a&&(a=O(a),n=L),S.call(t,n,function(t,n){var r;switch(n.charAt(0)){case"$":return"$";case"&":return o;case"`":return u.slice(0,c);case"'":return u.slice(s);case"<":r=a[n.slice(1,-1)];break;default:var e=+n;if(0==e)return t;if(l<e){var i=I(e/10);return 0!==i&&i<=l?void 0===f[i-1]?n.charAt(1):f[i-1]+n.charAt(1):t}r=f[e-1]}return void 0===r?"":r})}(p,i,d,y,x,n);h<=d&&(l+=i.slice(h,d)+m,h=d+p.length)}return l+i.slice(h)}]})},function(t,n,r){"use strict";var f=r(2),a=r(192),s=r(128);r(122)("search",1,function(e,i,u,c){return[function(t){var n=e(this),r=null==t?void 0:t[i];return void 0!==r?r.call(t,n):new RegExp(t)[i](String(n))},function(t){var n=c(u,t,this);if(n.done)return n.value;var r=f(t),e=String(this),i=r.lastIndex;a(i,0)||(r.lastIndex=0);var o=s(r,e);return a(r.lastIndex,i)||(r.lastIndex=i),null===o?-1:o.index}]})},function(t,n,r){"use strict";var l=r(124),x=r(2),m=r(119),S=r(136),w=r(8),_=r(128),h=r(152),e=r(4),O=Math.min,v=[].push,u="split",p="length",d="lastIndex",E=4294967295,M=!e(function(){RegExp(E,"y")});r(122)("split",2,function(i,o,y,g){var b="c"=="abbc"[u](/(b)*/)[1]||4!="test"[u](/(?:)/,-1)[p]||2!="ab"[u](/(?:ab)*/)[p]||4!="."[u](/(.?)(.?)/)[p]||1<"."[u](/()()/)[p]||""[u](/.?/)[p]?function(t,n){var r=String(this);if(void 0===t&&0===n)return[];if(!l(t))return y.call(r,t,n);for(var e,i,o,u=[],c=(t.ignoreCase?"i":"")+(t.multiline?"m":"")+(t.unicode?"u":"")+(t.sticky?"y":""),f=0,a=void 0===n?E:n>>>0,s=new RegExp(t.source,c+"g");(e=h.call(s,r))&&!(f<(i=s[d])&&(u.push(r.slice(f,e.index)),1<e[p]&&e.index<r[p]&&v.apply(u,e.slice(1)),o=e[0][p],f=i,u[p]>=a));)s[d]===e.index&&s[d]++;return f===r[p]?!o&&s.test("")||u.push(""):u.push(r.slice(f)),u[p]>a?u.slice(0,a):u}:"0"[u](void 0,0)[p]?function(t,n){return void 0===t&&0===n?[]:y.call(this,t,n)}:y;return[function(t,n){var r=i(this),e=null==t?void 0:t[o];return void 0!==e?e.call(t,r,n):b.call(String(r),t,n)},function(t,n){var r=g(b,t,this,n,b!==y);if(r.done)return r.value;var e=x(t),i=String(this),o=m(e,RegExp),u=e.unicode,c=(e.ignoreCase?"i":"")+(e.multiline?"m":"")+(e.unicode?"u":"")+(M?"y":"g"),f=new o(M?e:"^(?:"+e.source+")",c),a=void 0===n?E:n>>>0;if(0==a)return[];if(0===i.length)return null===_(f,i)?[i]:[];for(var s=0,l=0,h=[];l<i.length;){f.lastIndex=M?l:0;var v,p=_(f,M?i:i.slice(l));if(null===p||(v=O(w(f.lastIndex+(M?0:l)),i.length))===s)l=S(i,l,u);else{if(h.push(i.slice(s,l)),h.length===a)return h;for(var d=1;d<=p.length-1;d++)if(h.push(p[d]),h.length===a)return h;l=s=v}}return h.push(i.slice(s)),h}]})},function(t,n,r){"use strict";function e(t){r(27)(RegExp.prototype,c,t,!0)}r(198);var i=r(2),o=r(115),u=r(10),c="toString",f=/./[c];r(4)(function(){return"/a/b"!=f.call({source:"a",flags:"b"})})?e(function(){var t=i(this);return"/".concat(t.source,"/","flags"in t?t.flags:!u&&t instanceof RegExp?o.call(t):void 0)}):f.name!=c&&e(function(){return f.call(this)})},function(t,n,r){"use strict";r(28)("anchor",function(n){return function(t){return n(this,"a","name",t)}})},function(t,n,r){"use strict";r(28)("big",function(t){return function(){return t(this,"big","","")}})},function(t,n,r){"use strict";r(28)("blink",function(t){return function(){return t(this,"blink","","")}})},function(t,n,r){"use strict";r(28)("bold",function(t){return function(){return t(this,"b","","")}})},function(t,n,r){"use strict";var e=r(1),i=r(131)(!1);e(e.P,"String",{codePointAt:function(t){return i(this,t)}})},function(t,n,r){"use strict";var e=r(1),u=r(8),c=r(155),f="endsWith",a=""[f];e(e.P+e.F*r(142)(f),"String",{endsWith:function(t){var n=c(this,t,f),r=1<arguments.length?arguments[1]:void 0,e=u(n.length),i=void 0===r?e:Math.min(u(r),e),o=String(t);return a?a.call(n,o,i):n.slice(i-o.length,i)===o}})},function(t,n,r){"use strict";r(28)("fixed",function(t){return function(){return t(this,"tt","","")}})},function(t,n,r){"use strict";r(28)("fontcolor",function(n){return function(t){return n(this,"font","color",t)}})},function(t,n,r){"use strict";r(28)("fontsize",function(n){return function(t){return n(this,"font","size",t)}})},function(t,n,r){var e=r(1),o=r(78),u=String.fromCharCode,i=String.fromCodePoint;e(e.S+e.F*(!!i&&1!=i.length),"String",{fromCodePoint:function(t){for(var n,r=[],e=arguments.length,i=0;i<e;){if(n=+arguments[i++],o(n,1114111)!==n)throw RangeError(n+" is not a valid code point");r.push(n<65536?u(n):u(55296+((n-=65536)>>10),n%1024+56320))}return r.join("")}})},function(t,n,r){"use strict";var e=r(1),i=r(155);e(e.P+e.F*r(142)("includes"),"String",{includes:function(t){return!!~i(this,t,"includes").indexOf(t,1<arguments.length?arguments[1]:void 0)}})},function(t,n,r){"use strict";r(28)("italics",function(t){return function(){return t(this,"i","","")}})},function(t,n,r){"use strict";var e=r(131)(!0);r(147)(String,"String",function(t){this._t=String(t),this._i=0},function(){var t,n=this._t,r=this._i;return r>=n.length?{value:void 0,done:!0}:(t=e(n,r),this._i+=t.length,{value:t,done:!1})})},function(t,n,r){"use strict";r(28)("link",function(n){return function(t){return n(this,"a","href",t)}})},function(t,n,r){var e=r(1),u=r(33),c=r(8);e(e.S,"String",{raw:function(t){for(var n=u(t.raw),r=c(n.length),e=arguments.length,i=[],o=0;o<r;)i.push(String(n[o++])),o<e&&i.push(String(arguments[o]));return i.join("")}})},function(t,n,r){var e=r(1);e(e.P,"String",{repeat:r(156)})},function(t,n,r){"use strict";r(28)("small",function(t){return function(){return t(this,"small","","")}})},function(t,n,r){"use strict";var e=r(1),i=r(8),o=r(155),u="startsWith",c=""[u];e(e.P+e.F*r(142)(u),"String",{startsWith:function(t){var n=o(this,t,u),r=i(Math.min(1<arguments.length?arguments[1]:void 0,n.length)),e=String(t);return c?c.call(n,e,r):n.slice(r,r+e.length)===e}})},function(t,n,r){"use strict";r(28)("strike",function(t){return function(){return t(this,"strike","","")}})},function(t,n,r){"use strict";r(28)("sub",function(t){return function(){return t(this,"sub","","")}})},function(t,n,r){"use strict";r(28)("sup",function(t){return function(){return t(this,"sup","","")}})},function(t,n,r){"use strict";r(84)("trim",function(t){return function(){return t(this,3)}})},function(t,n,r){"use strict";function e(t){var n=H[t]=A(G[V]);return n._k=t,n}function i(t,n){O(t);for(var r,e=w(n=P(n)),i=0,o=e.length;i<o;)nt(t,r=e[i++],n[r]);return t}function o(t){var n=z.call(this,t=j(t,!0));return!(this===$&&s(H,t)&&!s(J,t))&&(!(n||!s(this,t)||!s(H,t)||s(this,B)&&this[B][t])||n)}function u(t,n){if(t=P(t),n=j(n,!0),t!==$||!s(H,n)||s(J,n)){var r=R(t,n);return!r||!s(H,n)||s(t,B)&&t[B][n]||(r.enumerable=!0),r}}function c(t){for(var n,r=D(P(t)),e=[],i=0;r.length>i;)s(H,n=r[i++])||n==B||n==p||e.push(n);return e}function f(t){for(var n,r=t===$,e=D(r?J:P(t)),i=[],o=0;e.length>o;)!s(H,n=e[o++])||r&&!s($,n)||i.push(H[n]);return i}var a=r(3),s=r(30),l=r(10),h=r(1),v=r(27),p=r(69).KEY,d=r(4),y=r(118),g=r(83),b=r(79),x=r(7),m=r(195),S=r(160),w=r(218),_=r(123),O=r(2),E=r(5),M=r(17),P=r(33),j=r(53),F=r(75),A=r(72),I=r(184),L=r(31),N=r(127),T=r(11),k=r(74),R=L.f,C=T.f,D=I.f,G=a.Symbol,W=a.JSON,U=W&&W.stringify,V="prototype",B=x("_hidden"),q=x("toPrimitive"),z={}.propertyIsEnumerable,K=y("symbol-registry"),H=y("symbols"),J=y("op-symbols"),$=Object[V],Y="function"==typeof G&&!!N.f,X=a.QObject,Q=!X||!X[V]||!X[V].findChild,Z=l&&d(function(){return 7!=A(C({},"a",{get:function(){return C(this,"a",{value:7}).a}})).a})?function(t,n,r){var e=R($,n);e&&delete $[n],C(t,n,r),e&&t!==$&&C($,n,e)}:C,tt=Y&&"symbol"==typeof G.iterator?function(t){return"symbol"==typeof t}:function(t){return t instanceof G},nt=function(t,n,r){return t===$&&nt(J,n,r),O(t),n=j(n,!0),O(r),s(H,n)?(r.enumerable?(s(t,B)&&t[B][n]&&(t[B][n]=!1),r=A(r,{enumerable:F(0,!1)})):(s(t,B)||C(t,B,F(1,{})),t[B][n]=!0),Z(t,n,r)):C(t,n,r)};Y||(v((G=function(){if(this instanceof G)throw TypeError("Symbol is not a constructor!");var n=b(0<arguments.length?arguments[0]:void 0),r=function(t){this===$&&r.call(J,t),s(this,B)&&s(this[B],n)&&(this[B][n]=!1),Z(this,n,F(1,t))};return l&&Q&&Z($,n,{configurable:!0,set:r}),e(n)})[V],"toString",function(){return this._k}),L.f=u,T.f=nt,r(73).f=I.f=c,r(117).f=o,N.f=f,l&&!r(68)&&v($,"propertyIsEnumerable",o,!0),m.f=function(t){return e(x(t))}),h(h.G+h.W+h.F*!Y,{Symbol:G});for(var rt="hasInstance,isConcatSpreadable,iterator,match,replace,search,species,split,toPrimitive,toStringTag,unscopables".split(","),et=0;rt.length>et;)x(rt[et++]);for(var it=k(x.store),ot=0;it.length>ot;)S(it[ot++]);h(h.S+h.F*!Y,"Symbol",{for:function(t){return s(K,t+="")?K[t]:K[t]=G(t)},keyFor:function(t){if(!tt(t))throw TypeError(t+" is not a symbol!");for(var n in K)if(K[n]===t)return n},useSetter:function(){Q=!0},useSimple:function(){Q=!1}}),h(h.S+h.F*!Y,"Object",{create:function(t,n){return void 0===n?A(t):i(A(t),n)},defineProperty:nt,defineProperties:i,getOwnPropertyDescriptor:u,getOwnPropertyNames:c,getOwnPropertySymbols:f});var ut=d(function(){N.f(1)});h(h.S+h.F*ut,"Object",{getOwnPropertySymbols:function(t){return N.f(M(t))}}),W&&h(h.S+h.F*(!Y||d(function(){var t=G();return"[null]"!=U([t])||"{}"!=U({a:t})||"{}"!=U(Object(t))})),"JSON",{stringify:function(t){for(var n,r,e=[t],i=1;i<arguments.length;)e.push(arguments[i++]);if(r=n=e[1],(E(n)||void 0!==t)&&!tt(t))return _(n)||(n=function(t,n){if("function"==typeof r&&(n=r.call(this,t,n)),!tt(n))return n}),e[1]=n,U.apply(W,e)}}),G[V][q]||r(26)(G[V],q,G[V].valueOf),g(G,"Symbol"),g(Math,"Math",!0),g(a.JSON,"JSON",!0)},function(t,n,r){"use strict";var e=r(1),i=r(132),o=r(159),a=r(2),s=r(78),l=r(8),u=r(5),c=r(3).ArrayBuffer,h=r(119),v=o.ArrayBuffer,p=o.DataView,f=i.ABV&&c.isView,d=v.prototype.slice,y=i.VIEW,g="ArrayBuffer";e(e.G+e.W+e.F*(c!==v),{ArrayBuffer:v}),e(e.S+e.F*!i.CONSTR,g,{isView:function(t){return f&&f(t)||u(t)&&y in t}}),e(e.P+e.U+e.F*r(4)(function(){return!new v(2).slice(1,void 0).byteLength}),g,{slice:function(t,n){if(void 0!==d&&void 0===n)return d.call(a(this),t);for(var r=a(this).byteLength,e=s(t,r),i=s(void 0===n?r:n,r),o=new(h(this,v))(l(i-e)),u=new p(this),c=new p(o),f=0;e<i;)c.setUint8(f++,u.getUint8(e++));return o}}),r(77)(g)},function(t,n,r){var e=r(1);e(e.G+e.W+e.F*!r(132).ABV,{DataView:r(159).DataView})},function(t,n,r){r(57)("Float32",4,function(e){return function(t,n,r){return e(this,t,n,r)}})},function(t,n,r){r(57)("Float64",8,function(e){return function(t,n,r){return e(this,t,n,r)}})},function(t,n,r){r(57)("Int16",2,function(e){return function(t,n,r){return e(this,t,n,r)}})},function(t,n,r){r(57)("Int32",4,function(e){return function(t,n,r){return e(this,t,n,r)}})},function(t,n,r){r(57)("Int8",1,function(e){return function(t,n,r){return e(this,t,n,r)}})},function(t,n,r){r(57)("Uint16",2,function(e){return function(t,n,r){return e(this,t,n,r)}})},function(t,n,r){r(57)("Uint32",4,function(e){return function(t,n,r){return e(this,t,n,r)}})},function(t,n,r){r(57)("Uint8",1,function(e){return function(t,n,r){return e(this,t,n,r)}})},function(t,n,r){r(57)("Uint8",1,function(e){return function(t,n,r){return e(this,t,n,r)}},!0)},function(t,n,r){"use strict";var e=r(172),i=r(80);r(121)("WeakSet",function(t){return function(){return t(this,0<arguments.length?arguments[0]:void 0)}},{add:function(t){return e.def(i(this,"WeakSet"),t,!0)}},e,!1,!0)},function(t,n,r){"use strict";var e=r(1),i=r(173),o=r(17),u=r(8),c=r(21),f=r(138);e(e.P,"Array",{flatMap:function(t){var n,r,e=o(this);return c(t),n=u(e.length),r=f(e,0),i(r,e,e,n,0,1,t,arguments[1]),r}}),r(67)("flatMap")},function(t,n,r){"use strict";var e=r(1),i=r(173),o=r(17),u=r(8),c=r(49),f=r(138);e(e.P,"Array",{flatten:function(){var t=arguments[0],n=o(this),r=u(n.length),e=f(n,0);return i(e,n,n,r,0,void 0===t?1:c(t)),e}}),r(67)("flatten")},function(t,n,r){"use strict";var e=r(1),i=r(120)(!0);e(e.P,"Array",{includes:function(t){return i(this,t,1<arguments.length?arguments[1]:void 0)}}),r(67)("includes")},function(t,n,r){var e=r(1),i=r(150)(),o=r(3).process,u="process"==r(45)(o);e(e.G,{asap:function(t){var n=u&&o.domain;i(n?n.bind(t):t)}})},function(t,n,r){var e=r(1),i=r(45);e(e.S,"Error",{isError:function(t){return"Error"===i(t)}})},function(t,n,r){var e=r(1);e(e.G,{global:r(3)})},function(t,n,r){r(129)("Map")},function(t,n,r){r(130)("Map")},function(t,n,r){var e=r(1);e(e.P+e.R,"Map",{toJSON:r(171)("Map")})},function(t,n,r){var e=r(1);e(e.S,"Math",{clamp:function(t,n,r){return Math.min(r,Math.max(n,t))}})},function(t,n,r){var e=r(1);e(e.S,"Math",{DEG_PER_RAD:Math.PI/180})},function(t,n,r){var e=r(1),i=180/Math.PI;e(e.S,"Math",{degrees:function(t){return t*i}})},function(t,n,r){var e=r(1),o=r(181),u=r(179);e(e.S,"Math",{fscale:function(t,n,r,e,i){return u(o(t,n,r,e,i))}})},function(t,n,r){var e=r(1);e(e.S,"Math",{iaddh:function(t,n,r,e){var i=t>>>0,o=r>>>0;return(n>>>0)+(e>>>0)+((i&o|(i|o)&~(i+o>>>0))>>>31)|0}})},function(t,n,r){var e=r(1);e(e.S,"Math",{imulh:function(t,n){var r=+t,e=+n,i=65535&r,o=65535&e,u=r>>16,c=e>>16,f=(u*o>>>0)+(i*o>>>16);return u*c+(f>>16)+((i*c>>>0)+(65535&f)>>16)}})},function(t,n,r){var e=r(1);e(e.S,"Math",{isubh:function(t,n,r,e){var i=t>>>0,o=r>>>0;return(n>>>0)-(e>>>0)-((~i&o|~(i^o)&i-o>>>0)>>>31)|0}})},function(t,n,r){var e=r(1);e(e.S,"Math",{RAD_PER_DEG:180/Math.PI})},function(t,n,r){var e=r(1),i=Math.PI/180;e(e.S,"Math",{radians:function(t){return t*i}})},function(t,n,r){var e=r(1);e(e.S,"Math",{scale:r(181)})},function(t,n,r){var e=r(1);e(e.S,"Math",{signbit:function(t){return(t=+t)!=t?t:0==t?1/t==1/0:0<t}})},function(t,n,r){var e=r(1);e(e.S,"Math",{umulh:function(t,n){var r=+t,e=+n,i=65535&r,o=65535&e,u=r>>>16,c=e>>>16,f=(u*o>>>0)+(i*o>>>16);return u*c+(f>>>16)+((i*c>>>0)+(65535&f)>>>16)}})},function(t,n,r){"use strict";var e=r(1),i=r(17),o=r(21),u=r(11);r(10)&&e(e.P+r(126),"Object",{__defineGetter__:function(t,n){u.f(i(this),t,{get:o(n),enumerable:!0,configurable:!0})}})},function(t,n,r){"use strict";var e=r(1),i=r(17),o=r(21),u=r(11);r(10)&&e(e.P+r(126),"Object",{__defineSetter__:function(t,n){u.f(i(this),t,{set:o(n),enumerable:!0,configurable:!0})}})},function(t,n,r){var e=r(1),i=r(186)(!0);e(e.S,"Object",{entries:function(t){return i(t)}})},function(t,n,r){var e=r(1),f=r(187),a=r(33),s=r(31),l=r(139);e(e.S,"Object",{getOwnPropertyDescriptors:function(t){for(var n,r,e=a(t),i=s.f,o=f(e),u={},c=0;o.length>c;)void 0!==(r=i(e,n=o[c++]))&&l(u,n,r);return u}})},function(t,n,r){"use strict";var e=r(1),i=r(17),o=r(53),u=r(32),c=r(31).f;r(10)&&e(e.P+r(126),"Object",{__lookupGetter__:function(t){var n,r=i(this),e=o(t,!0);do{if(n=c(r,e))return n.get}while(r=u(r))}})},function(t,n,r){"use strict";var e=r(1),i=r(17),o=r(53),u=r(32),c=r(31).f;r(10)&&e(e.P+r(126),"Object",{__lookupSetter__:function(t){var n,r=i(this),e=o(t,!0);do{if(n=c(r,e))return n.set}while(r=u(r))}})},function(t,n,r){var e=r(1),i=r(186)(!1);e(e.S,"Object",{values:function(t){return i(t)}})},function(t,n,r){"use strict";function i(t){return null==t?void 0:v(t)}function o(t){var n=t._c;n&&(t._c=void 0,n())}function u(t){return void 0===t._o}function c(t){u(t)||(t._o=void 0,o(t))}function e(t,n){p(t),this._c=void 0,this._o=t,t=new m(this);try{var r=n(t),e=r;null!=r&&("function"==typeof r.unsubscribe?r=function(){e.unsubscribe()}:v(r),this._c=r)}catch(n){return void t.error(n)}u(this)&&o(this)}var f=r(1),a=r(3),s=r(46),l=r(150)(),h=r(7)("observable"),v=r(21),p=r(2),d=r(70),y=r(76),g=r(26),b=r(71),x=b.RETURN;e.prototype=y({},{unsubscribe:function(){c(this)}});var m=function(t){this._s=t};m.prototype=y({},{next:function(t){var n=this._s;if(!u(n)){var r=n._o;try{var e=i(r.next);if(e)return e.call(r,t)}catch(t){try{c(n)}finally{throw t}}}},error:function(t){var n=this._s;if(u(n))throw t;var r=n._o;n._o=void 0;try{var e=i(r.error);if(!e)throw t;t=e.call(r,t)}catch(t){try{o(n)}finally{throw t}}return o(n),t},complete:function(t){var n=this._s;if(!u(n)){var r=n._o;n._o=void 0;try{var e=i(r.complete);t=e?e.call(r,t):void 0}catch(t){try{o(n)}finally{throw t}}return o(n),t}}});var S=function(t){d(this,S,"Observable","_f")._f=v(t)};y(S.prototype,{subscribe:function(t){return new e(t,this._f)},forEach:function(e){var i=this;return new(s.Promise||a.Promise)(function(t,n){v(e);var r=i.subscribe({next:function(t){try{return e(t)}catch(t){n(t),r.unsubscribe()}},error:n,complete:t})})}}),y(S,{from:function(t){var n="function"==typeof this?this:S,r=i(p(t)[h]);if(r){var e=p(r.call(t));return e.constructor===n?e:new n(function(t){return e.subscribe(t)})}return new n(function(n){var r=!1;return l(function(){if(!r){try{if(b(t,!1,function(t){if(n.next(t),r)return x})===x)return}catch(t){if(r)throw t;return void n.error(t)}n.complete()}}),function(){r=!0}})},of:function(){for(var t=0,n=arguments.length,e=new Array(n);t<n;)e[t]=arguments[t++];return new("function"==typeof this?this:S)(function(n){var r=!1;return l(function(){if(!r){for(var t=0;t<e.length;++t)if(n.next(e[t]),r)return;n.complete()}}),function(){r=!0}})}}),g(S.prototype,h,function(){return this}),f(f.G,{Observable:S}),r(77)("Observable")},function(t,n,r){"use strict";var e=r(1),i=r(46),o=r(3),u=r(119),c=r(191);e(e.P+e.R,"Promise",{finally:function(n){var r=u(this,i.Promise||o.Promise),t="function"==typeof n;return this.then(t?function(t){return c(r,n()).then(function(){return t})}:n,t?function(t){return c(r,n()).then(function(){throw t})}:n)}})},function(t,n,r){"use strict";var e=r(1),i=r(151),o=r(190);e(e.S,"Promise",{try:function(t){var n=i.f(this),r=o(t);return(r.e?n.reject:n.resolve)(r.v),n.promise}})},function(t,n,r){var e=r(56),i=r(2),o=e.key,u=e.set;e.exp({defineMetadata:function(t,n,r,e){u(t,n,i(r),o(e))}})},function(t,n,r){var e=r(56),o=r(2),u=e.key,c=e.map,f=e.store;e.exp({deleteMetadata:function(t,n){var r=arguments.length<3?void 0:u(arguments[2]),e=c(o(n),r,!1);if(void 0===e||!e.delete(t))return!1;if(e.size)return!0;var i=f.get(n);return i.delete(r),!!i.size||f.delete(n)}})},function(t,n,r){var o=r(199),u=r(167),e=r(56),i=r(2),c=r(32),f=e.keys,a=e.key,s=function(t,n){var r=f(t,n),e=c(t);if(null===e)return r;var i=s(e,n);return i.length?r.length?u(new o(r.concat(i))):i:r};e.exp({getMetadataKeys:function(t){return s(i(t),arguments.length<2?void 0:a(arguments[1]))}})},function(t,n,r){var e=r(56),i=r(2),o=r(32),u=e.has,c=e.get,f=e.key,a=function(t,n,r){if(u(t,n,r))return c(t,n,r);var e=o(n);return null!==e?a(t,e,r):void 0};e.exp({getMetadata:function(t,n){return a(t,i(n),arguments.length<3?void 0:f(arguments[2]))}})},function(t,n,r){var e=r(56),i=r(2),o=e.keys,u=e.key;e.exp({getOwnMetadataKeys:function(t){return o(i(t),arguments.length<2?void 0:u(arguments[1]))}})},function(t,n,r){var e=r(56),i=r(2),o=e.get,u=e.key;e.exp({getOwnMetadata:function(t,n){return o(t,i(n),arguments.length<3?void 0:u(arguments[2]))}})},function(t,n,r){var e=r(56),i=r(2),o=r(32),u=e.has,c=e.key,f=function(t,n,r){if(u(t,n,r))return!0;var e=o(n);return null!==e&&f(t,e,r)};e.exp({hasMetadata:function(t,n){return f(t,i(n),arguments.length<3?void 0:c(arguments[2]))}})},function(t,n,r){var e=r(56),i=r(2),o=e.has,u=e.key;e.exp({hasOwnMetadata:function(t,n){return o(t,i(n),arguments.length<3?void 0:u(arguments[2]))}})},function(t,n,r){var e=r(56),i=r(2),o=r(21),u=e.key,c=e.set;e.exp({metadata:function(r,e){return function(t,n){c(r,e,(void 0!==n?i:o)(t),u(n))}}})},function(t,n,r){r(129)("Set")},function(t,n,r){r(130)("Set")},function(t,n,r){var e=r(1);e(e.P+e.R,"Set",{toJSON:r(171)("Set")})},function(t,n,r){"use strict";var e=r(1),i=r(131)(!0);e(e.P,"String",{at:function(t){return i(this,t)}})},function(t,n,r){"use strict";function i(t,n){this._r=t,this._s=n}var e=r(1),o=r(51),u=r(8),c=r(124),f=r(115),a=RegExp.prototype;r(146)(i,"RegExp String",function(){var t=this._r.exec(this._s);return{value:t,done:null===t}}),e(e.P,"String",{matchAll:function(t){if(o(this),!c(t))throw TypeError(t+" is not a regexp!");var n=String(this),r="flags"in a?String(t.flags):f.call(t),e=new RegExp(t.source,~r.indexOf("g")?r:"g"+r);return e.lastIndex=u(t.lastIndex),new i(e,n)}})},function(t,n,r){"use strict";var e=r(1),i=r(193),o=r(133),u=/Version\/10\.\d+(\.\d+)?( Mobile\/\w+)? Safari\//.test(o);e(e.P+e.F*u,"String",{padEnd:function(t){return i(this,t,1<arguments.length?arguments[1]:void 0,!1)}})},function(t,n,r){"use strict";var e=r(1),i=r(193),o=r(133),u=/Version\/10\.\d+(\.\d+)?( Mobile\/\w+)? Safari\//.test(o);e(e.P+e.F*u,"String",{padStart:function(t){return i(this,t,1<arguments.length?arguments[1]:void 0,!0)}})},function(t,n,r){"use strict";r(84)("trimLeft",function(t){return function(){return t(this,1)}},"trimStart")},function(t,n,r){"use strict";r(84)("trimRight",function(t){return function(){return t(this,2)}},"trimEnd")},function(t,n,r){r(160)("asyncIterator")},function(t,n,r){r(160)("observable")},function(t,n,r){var e=r(1);e(e.S,"System",{global:r(3)})},function(t,n,r){r(129)("WeakMap")},function(t,n,r){r(130)("WeakMap")},function(t,n,r){r(129)("WeakSet")},function(t,n,r){r(130)("WeakSet")},function(t,n,r){for(var e=r(162),i=r(74),o=r(27),u=r(3),c=r(26),f=r(82),a=r(7),s=a("iterator"),l=a("toStringTag"),h=f.Array,v={CSSRuleList:!0,CSSStyleDeclaration:!1,CSSValueList:!1,ClientRectList:!1,DOMRectList:!1,DOMStringList:!1,DOMTokenList:!0,DataTransferItemList:!1,FileList:!1,HTMLAllCollection:!1,HTMLCollection:!1,HTMLFormElement:!1,HTMLSelectElement:!1,MediaList:!0,MimeTypeArray:!1,NamedNodeMap:!1,NodeList:!0,PaintRequestList:!1,Plugin:!1,PluginArray:!1,SVGLengthList:!1,SVGNumberList:!1,SVGPathSegList:!1,SVGPointList:!1,SVGStringList:!1,SVGTransformList:!1,SourceBufferList:!1,StyleSheetList:!0,TextTrackCueList:!1,TextTrackList:!1,TouchList:!1},p=i(v),d=0;d<p.length;d++){var y,g=p[d],b=v[g],x=u[g],m=x&&x.prototype;if(m&&(m[s]||c(m,s,h),m[l]||c(m,l,g),f[g]=h,b))for(y in e)m[y]||o(m,y,e[y],!0)}},function(t,n,r){var e=r(1),i=r(158);e(e.G+e.B,{setImmediate:i.set,clearImmediate:i.clear})},function(t,n,r){function e(i){return function(t,n){var r=2<arguments.length,e=!!r&&c.call(arguments,2);return i(r?function(){("function"==typeof t?t:Function(t)).apply(this,e)}:t,n)}}var i=r(3),o=r(1),u=r(133),c=[].slice,f=/MSIE .\./.test(u);o(o.G+o.B+o.F*f,{setTimeout:e(i.setTimeout),setInterval:e(i.setInterval)})},function(t,n,r){r(341),r(280),r(282),r(281),r(284),r(286),r(291),r(285),r(283),r(293),r(292),r(288),r(289),r(287),r(279),r(290),r(294),r(295),r(247),r(249),r(248),r(297),r(296),r(267),r(277),r(278),r(268),r(269),r(270),r(271),r(272),r(273),r(274),r(275),r(276),r(250),r(251),r(252),r(253),r(254),r(255),r(256),r(257),r(258),r(259),r(260),r(261),r(262),r(263),r(264),r(265),r(266),r(328),r(333),r(340),r(331),r(323),r(324),r(329),r(334),r(336),r(319),r(320),r(321),r(322),r(325),r(326),r(327),r(330),r(332),r(335),r(337),r(338),r(339),r(242),r(244),r(243),r(246),r(245),r(231),r(229),r(235),r(232),r(238),r(240),r(228),r(234),r(225),r(239),r(223),r(237),r(236),r(230),r(233),r(222),r(224),r(227),r(226),r(241),r(162),r(313),r(197),r(318),r(198),r(314),r(315),r(316),r(317),r(298),r(196),r(199),r(200),r(353),r(342),r(343),r(348),r(351),r(352),r(346),r(349),r(347),r(350),r(344),r(345),r(299),r(300),r(301),r(302),r(303),r(306),r(304),r(305),r(307),r(308),r(309),r(310),r(312),r(311),r(356),r(354),r(355),r(397),r(400),r(399),r(401),r(402),r(398),r(403),r(404),r(378),r(381),r(377),r(375),r(376),r(379),r(380),r(362),r(396),r(361),r(395),r(407),r(409),r(360),r(394),r(406),r(408),r(359),r(405),r(358),r(363),r(364),r(365),r(366),r(367),r(369),r(368),r(370),r(371),r(372),r(374),r(373),r(383),r(384),r(385),r(386),r(388),r(387),r(390),r(389),r(391),r(392),r(393),r(357),r(382),r(412),r(411),r(410),t.exports=r(46)},function(t,n){t.exports=function(t,n){if("string"==typeof n)return t.insertAdjacentHTML("afterend",n);var r=t.nextSibling;return r?t.parentNode.insertBefore(n,r):t.parentNode.appendChild(n)}}])</script><script src="/./main.b8fa34.js"></script><script>!function(){var e=document.createElement("script");document.getElementsByTagName("body")[0].appendChild(e),e.setAttribute("src","/slider.b6a41a.js")}()</script><div class="tools-col" q-class="show:isShow,hide:isShow|isFalse" q-on="click:stop(e)"><div class="tools-nav header-menu"><ul style="width:70%"><li style="width:33.333333333333336%" q-on="click: openSlider(e, 'innerArchive')"><a href="javascript:void(0)" q-class="active:innerArchive">所有文章</a></li><li style="width:33.333333333333336%" q-on="click: openSlider(e, 'friends')"><a href="javascript:void(0)" q-class="active:friends">友情链接</a></li><li style="width:33.333333333333336%" q-on="click: openSlider(e, 'aboutme')"><a href="javascript:void(0)" q-class="active:aboutme">关于我</a></li></ul></div><div class="tools-wrap"><section class="tools-section tools-section-all" q-show="innerArchive"><div class="search-wrap"><input class="search-ipt" q-model="search" type="text" placeholder="find something…"> <i class="icon-search icon" q-show="search|isEmptyStr"></i> <i class="icon-close icon" q-show="search|isNotEmptyStr" q-on="click:clearChose(e)"></i></div><div class="widget tagcloud search-tag"><p class="search-tag-wording">tag:</p><label class="search-switch"><input type="checkbox" q-on="click:toggleTag(e)" q-attr="checked:showTags"></label><ul class="article-tag-list" q-show="showTags"><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color3">随笔</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color3">安卓</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color2">gradle</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color5">软件使用</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color4">Git</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color4">anaconda</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color5">函数指针</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color4">嵌入式</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color4">IIC</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color3">IMX6ULL</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color5">学习笔记</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color2">Typora</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color4">Bug</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color1">Linux</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color5">Word</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color1">Visio</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color4">BUG</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color5">hexo</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color4">ros</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color4">SSH</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color3">aplayer</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color1">正则表达式</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color3">RE</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color3">递归</a></li><li class="article-tag-list-item"><a href="javascript:void(0)" class="js-tag color5">编程语言</a></li><div class="clearfix"></div></ul></div><ul class="search-ul"><p q-show="jsonFail" style="padding:20px;font-size:12px">缺失模块。<br>1、请确保node版本大于6.2<br>2、在博客根目录(注意不是yilia-plus根目录)执行以下命令:<br>npm i hexo-generator-json-content --save<br><br>3、在根目录_config.yml里添加配置:<pre style="font-size:12px" q-show="jsonFail">
jsonContent:
meta: false
pages: false
posts:
title: true
date: true
path: true
text: false
raw: false
content: false
slug: false
updated: false
comments: false
link: false
permalink: false
excerpt: false
categories: false
tags: true
</pre></p><li class="search-li" q-repeat="items" q-show="isShow"><a q-attr="href:path|urlformat" class="search-title"><i class="icon-quo-left icon"></i><span q-text="title"></span></a><p class="search-time"><i class="icon-calendar icon"></i> <span q-text="date|dateformat"></span></p><p class="search-tag"><i class="icon-price-tags icon"></i> <span q-repeat="tags" q-on="click:choseTag(e, name)" q-text="name|tagformat"></span></p></li></ul></section><section class="tools-section tools-section-friends" q-show="friends"><ul class="search-ul"><li class="search-li"><a href="https://blog.csdn.net/qq_37000789" target="_blank" class="search-title" title="个人博客"><i class="icon-link icon"></i> 正月点灯笼的博客</a></li><li class="search-li"><a href="https://blog.csdn.net/ld1461575230?spm=1001.2101.3001.5343&type=blog" target="_blank" class="search-title" title="个人博客"><i class="icon-link icon"></i> 哈哈哈的嘎嘎嘎的博客</a></li><li class="search-li"><a href="https://gujiakai.gitee.io/" target="_blank" class="search-title" title="个人博客"><i class="icon-link icon"></i> 摸鱼的小灰栗的博客</a></li></ul></section><section class="tools-section tools-section-me" q-show="aboutme"><div class="aboutme-wrap" id="aboutme">熟悉的技术:<br>C/C++、Python、嵌入式、ROS。<br>开源爱好者、Linux爱好者。<br>欢迎访问我的博客。hi✿(。◕ᴗ◕。)✿</div></section></div></div><div class="pswp" tabindex="-1" role="dialog" aria-hidden="true"><div class="pswp__bg"></div><div class="pswp__scroll-wrap"><div class="pswp__container"><div class="pswp__item"></div><div class="pswp__item"></div><div class="pswp__item"></div></div><div class="pswp__ui pswp__ui--hidden"><div class="pswp__top-bar"><div class="pswp__counter"></div><button class="pswp__button pswp__button--close" title="Close (Esc)"></button> <button class="pswp__button pswp__button--share" style="display:none" title="Share"></button> <button class="pswp__button pswp__button--fs" title="Toggle fullscreen"></button> <button class="pswp__button pswp__button--zoom" title="Zoom in/out"></button><div class="pswp__preloader"><div class="pswp__preloader__icn"><div class="pswp__preloader__cut"><div class="pswp__preloader__donut"></div></div></div></div></div><div class="pswp__share-modal pswp__share-modal--hidden pswp__single-tap"><div class="pswp__share-tooltip"></div></div><button class="pswp__button pswp__button--arrow--left" title="Previous (arrow left)"></button> <button class="pswp__button pswp__button--arrow--right" title="Next (arrow right)"></button><div class="pswp__caption"><div class="pswp__caption__center"></div></div></div></div></div></div><script type="text/javascript" src="/plugins/TweenMax/TweenMax.min.js"></script><script type="text/javascript" src="/plugins/TweenMax/HoverImg.js"></script><script>var hoverEffect = {
"subNav": ".left-col .social a"
}
for (var x in hoverEffect) {
// 初始化data-img属性值
let $em = document.querySelectorAll(hoverEffect[x]);
$em.forEach(($a) => {
//console.log($a.href);
if (isImageType($a.href)) {
$a.setAttribute("data-img", $a.href)
}
})
// 实现悬停元素预览图片效果
let $a = document.querySelectorAll(hoverEffect[x] + "[data-img]");
$a.forEach(($em) => {
new HoverImgFx1($em)
})
}</script><script type="text/javascript" src="/plugins/activate-power-mode/activate-power-mode.js"></script><script>POWERMODE.colorful=!0,POWERMODE.shake=!1,document.body.addEventListener("input",POWERMODE)</script><script type="text/javascript" src="/plugins/live2d-widget.js/L2Dwidget.min.js"></script><script type="text/javascript">var pluginModelPath="assets/",pluginRootPath="/live2d_models/",modelPathJson={epsilon2_1:"Epsilon2.1",gf:"Gantzert_Felixander","haru/01":"haru01","haru/02":"haru02",nietzsche:"nietzche"},modelName="hijiki",modelJsonPath="",modelJsonPath=modelPathJson[modelName]?pluginRootPath+modelName+"/"+pluginModelPath+modelPathJson[modelName]+".model.json":pluginRootPath+modelName+"/"+pluginModelPath+modelName+".model.json";L2Dwidget.init({model:{jsonPath:modelJsonPath},display:{position:"right",width:150,height:300,hOffset:10,vOffset:400},mobile:{show:!1,scale:.6},react:{opacity:.7}})</script></body></html>