Mercurial > cortex
changeset 449:09b7c8dd4365
first chapter done, half of last chapter done.
author | Robert McIntyre <rlm@mit.edu> |
---|---|
date | Wed, 26 Mar 2014 02:42:01 -0400 (2014-03-26) |
parents | af13fc73e851 |
children | 432f2c4646cb |
files | org/worm_learn.clj thesis/cortex.org thesis/images/basic-worm-view.png thesis/images/full-hand.png thesis/images/worm-identify-init.png thesis/rlm-cortex-meng.tex |
diffstat | 6 files changed, 455 insertions(+), 81 deletions(-) [+] |
line wrap: on
line diff
1.1 --- a/org/worm_learn.clj Tue Mar 25 22:54:41 2014 -0400 1.2 +++ b/org/worm_learn.clj Wed Mar 26 02:42:01 2014 -0400 1.3 @@ -27,6 +27,13 @@ 1.4 (defn worm-model [] 1.5 (load-blender-model "Models/worm/worm.blend")) 1.6 1.7 +(defn worm [] 1.8 + (let [model (load-blender-model "Models/worm/worm.blend")] 1.9 + {:body (doto model (body!)) 1.10 + :touch (touch! model) 1.11 + :proprioception (proprioception! model) 1.12 + :muscles (movement! model)})) 1.13 + 1.14 (def output-base (File. "/home/r/proj/cortex/render/worm-learn/curl")) 1.15 1.16 1.17 @@ -220,15 +227,8 @@ 1.18 (< 0.55 (contact worm-segment-top-tip head-touch)))))) 1.19 1.20 1.21 -(declare phi-space phi-scan) 1.22 +(declare phi-space phi-scan debug-experience) 1.23 1.24 -(defn debug-experience 1.25 - [experiences text] 1.26 - (cond 1.27 - (grand-circle? experiences) (.setText text "Grand Circle") 1.28 - (curled? experiences) (.setText text "Curled") 1.29 - (wiggling? experiences) (.setText text "Wiggling") 1.30 - (resting? experiences) (.setText text "Resting"))) 1.31 1.32 1.33 (def standard-world-view 1.34 @@ -277,16 +277,14 @@ 1.35 (.setFilterMode PssmShadowRenderer$FilterMode/Bilinear))] 1.36 (.addProcessor (.getViewPort world) pssm))) 1.37 1.38 +(defn debug-experience 1.39 + [experiences text] 1.40 + (cond 1.41 + (grand-circle? experiences) (.setText text "Grand Circle") 1.42 + (curled? experiences) (.setText text "Curled") 1.43 + (wiggling? experiences) (.setText text "Wiggling") 1.44 + (resting? experiences) (.setText text "Resting"))) 1.45 1.46 -(defn display-text [[x y :as location]] 1.47 - (let [] 1.48 - (.setLocalTranslation text 300 (.getLineHeight text) 0) 1.49 - (fn [world] 1.50 - 1.51 - 1.52 - 1.53 - 1.54 - (fn [new-text] 1.55 1.56 (defn worm-world 1.57 [& {:keys [record motor-control keybindings view experiences 1.58 @@ -294,14 +292,11 @@ 1.59 (let [{:keys [record motor-control keybindings view experiences 1.60 worm-model end-frame experience-watch]} 1.61 (merge (worm-world-defaults) settings) 1.62 - worm (doto (worm-model) (body!)) 1.63 - touch (touch! worm) 1.64 - prop (proprioception! worm) 1.65 - muscles (movement! worm) 1.66 - 1.67 + 1.68 touch-display (view-touch) 1.69 prop-display (view-proprioception) 1.70 muscle-display (view-movement) 1.71 + {:keys [proprioception touch muscles body]} (worm) 1.72 1.73 floor 1.74 (box 5 1 5 :position (Vector3f. 0 -10 0) 1.75 @@ -316,7 +311,7 @@ 1.76 (.setColor (ColorRGBA/Black)))] 1.77 1.78 (world 1.79 - (nodify [worm floor]) 1.80 + (nodify [body floor]) 1.81 (merge standard-debug-controls keybindings) 1.82 (fn [world] 1.83 (.setLocalTranslation 1.84 @@ -324,7 +319,7 @@ 1.85 (.attachChild (.getGuiNode world) worm-action) 1.86 1.87 (enable-good-shadows world) 1.88 - (.setShadowMode worm RenderQueue$ShadowMode/CastAndReceive) 1.89 + (.setShadowMode body RenderQueue$ShadowMode/CastAndReceive) 1.90 (.setShadowMode floor RenderQueue$ShadowMode/Receive) 1.91 1.92 (.setBackgroundColor (.getViewPort world) (ColorRGBA/White)) 1.93 @@ -332,7 +327,7 @@ 1.94 (.setDisplayFps world false) 1.95 (position-camera world view) 1.96 (.setTimer world timer) 1.97 - (display-dilated-time world timer) 1.98 + ;;(display-dilated-time world timer) 1.99 (when record 1.100 (dir! record) 1.101 (Capture/captureVideo 1.102 @@ -345,7 +340,7 @@ 1.103 (if (and end-frame (> (.getTime timer) end-frame)) 1.104 (.stop world)) 1.105 (let [muscle-data (vec (motor-control muscles)) 1.106 - proprioception-data (prop) 1.107 + proprioception-data (proprioception) 1.108 touch-data (mapv #(% (.getRootNode world)) touch)] 1.109 (when experiences 1.110 (record-experience!
2.1 --- a/thesis/cortex.org Tue Mar 25 22:54:41 2014 -0400 2.2 +++ b/thesis/cortex.org Wed Mar 26 02:42:01 2014 -0400 2.3 @@ -226,7 +226,7 @@ 2.4 #+end_listing 2.5 2.6 2.7 -** =CORTEX= is a toolkit for building sensate creatures 2.8 +** =CORTEX= is a toolkit for building sensate creatures 2.9 2.10 I built =CORTEX= to be a general AI research platform for doing 2.11 experiments involving multiple rich senses and a wide variety and 2.12 @@ -269,14 +269,16 @@ 2.13 engine designed to create cross-platform 3D desktop games. =CORTEX= 2.14 is mainly written in clojure, a dialect of =LISP= that runs on the 2.15 java virtual machine (JVM). The API for creating and simulating 2.16 - creatures is entirely expressed in clojure. Hearing is implemented 2.17 - as a layer of clojure code on top of a layer of java code on top of 2.18 - a layer of =C++= code which implements a modified version of 2.19 - =OpenAL= to support multiple listeners. =CORTEX= is the only 2.20 - simulation environment that I know of that can support multiple 2.21 - entities that can each hear the world from their own perspective. 2.22 - Other senses also require a small layer of Java code. =CORTEX= also 2.23 - uses =bullet=, a physics simulator written in =C=. 2.24 + creatures and senses is entirely expressed in clojure, though many 2.25 + senses are implemented at the layer of jMonkeyEngine or below. For 2.26 + example, for the sense of hearing I use a layer of clojure code on 2.27 + top of a layer of java JNI bindings that drive a layer of =C++= 2.28 + code which implements a modified version of =OpenAL= to support 2.29 + multiple listeners. =CORTEX= is the only simulation environment 2.30 + that I know of that can support multiple entities that can each 2.31 + hear the world from their own perspective. Other senses also 2.32 + require a small layer of Java code. =CORTEX= also uses =bullet=, a 2.33 + physics simulator written in =C=. 2.34 2.35 #+caption: Here is the worm from above modeled in Blender, a free 2.36 #+caption: 3D-modeling program. Senses and joints are described 2.37 @@ -285,26 +287,46 @@ 2.38 #+ATTR_LaTeX: :width 12cm 2.39 [[./images/blender-worm.png]] 2.40 2.41 + Here are some thing I anticipate that =CORTEX= might be used for: 2.42 + 2.43 + - exploring new ideas about sensory integration 2.44 + - distributed communication among swarm creatures 2.45 + - self-learning using free exploration, 2.46 + - evolutionary algorithms involving creature construction 2.47 + - exploration of exoitic senses and effectors that are not possible 2.48 + in the real world (such as telekenisis or a semantic sense) 2.49 + - imagination using subworlds 2.50 + 2.51 During one test with =CORTEX=, I created 3,000 entities each with 2.52 their own independent senses and ran them all at only 1/80 real 2.53 time. In another test, I created a detailed model of my own hand, 2.54 equipped with a realistic distribution of touch (more sensitive at 2.55 the fingertips), as well as eyes and ears, and it ran at around 1/4 2.56 - real time. 2.57 + real time. 2.58 2.59 - #+caption: Here is the worm from above modeled in Blender, a free 2.60 - #+caption: 3D-modeling program. Senses and joints are described 2.61 - #+caption: using special nodes in Blender. 2.62 - #+name: worm-recognition-intro 2.63 - #+ATTR_LaTeX: :width 15cm 2.64 - [[./images/full-hand.png]] 2.65 - 2.66 - 2.67 - 2.68 + #+BEGIN_LaTeX 2.69 + \begin{sidewaysfigure} 2.70 + \includegraphics[width=9.5in]{images/full-hand.png} 2.71 + \caption{Here is the worm from above modeled in Blender, 2.72 + a free 3D-modeling program. Senses and joints are described 2.73 + using special nodes in Blender. The senses are displayed on 2.74 + the right, and the simulation is displayed on the left. Notice 2.75 + that the hand is curling its fingers, that it can see its own 2.76 + finger from the eye in its palm, and thta it can feel its own 2.77 + thumb touching its palm.} 2.78 + \end{sidewaysfigure} 2.79 + #+END_LaTeX 2.80 2.81 - 2.82 ** Contributions 2.83 2.84 + I built =CORTEX=, a comprehensive platform for embodied AI 2.85 + experiments. =CORTEX= many new features lacking in other systems, 2.86 + such as sound. It is easy to create new creatures using Blender, a 2.87 + free 3D modeling program. 2.88 + 2.89 + I built =EMPATH=, which uses =CORTEX= to identify the actions of a 2.90 + worm-like creature using a computational model of empathy. 2.91 + 2.92 * Building =CORTEX= 2.93 2.94 ** To explore embodiment, we need a world, body, and senses 2.95 @@ -331,52 +353,409 @@ 2.96 2.97 * Empathy in a simulated worm 2.98 2.99 + Here I develop a computational model of empathy, using =CORTEX= as a 2.100 + base. Empathy in this context is the ability to observe another 2.101 + creature and infer what sorts of sensations that creature is 2.102 + feeling. My empathy algorithm involves multiple phases. First is 2.103 + free-play, where the creature moves around and gains sensory 2.104 + experience. From this experience I construct a representation of the 2.105 + creature's sensory state space, which I call \Phi-space. Using 2.106 + \Phi-space, I construct an efficient function which takes the 2.107 + limited data that comes from observing another creature and enriches 2.108 + it full compliment of imagined sensory data. I can then use the 2.109 + imagined sensory data to recognize what the observed creature is 2.110 + doing and feeling, using straightforward embodied action predicates. 2.111 + This is all demonstrated with using a simple worm-like creature, and 2.112 + recognizing worm-actions based on limited data. 2.113 + 2.114 + #+caption: Here is the worm with which we will be working. 2.115 + #+caption: It is composed of 5 segments. Each segment has a 2.116 + #+caption: pair of extensor and flexor muscles. Each of the 2.117 + #+caption: worm's four joints is a hinge joint which allows 2.118 + #+caption: 30 degrees of rotation to either side. Each segment 2.119 + #+caption: of the worm is touch-capable and has a uniform 2.120 + #+caption: distribution of touch sensors on each of its faces. 2.121 + #+caption: Each joint has a proprioceptive sense to detect 2.122 + #+caption: relative positions. The worm segments are all the 2.123 + #+caption: same except for the first one, which has a much 2.124 + #+caption: higher weight than the others to allow for easy 2.125 + #+caption: manual motor control. 2.126 + #+name: basic-worm-view 2.127 + #+ATTR_LaTeX: :width 10cm 2.128 + [[./images/basic-worm-view.png]] 2.129 + 2.130 + #+caption: Program for reading a worm from a blender file and 2.131 + #+caption: outfitting it with the senses of proprioception, 2.132 + #+caption: touch, and the ability to move, as specified in the 2.133 + #+caption: blender file. 2.134 + #+name: get-worm 2.135 + #+begin_listing clojure 2.136 + #+begin_src clojure 2.137 +(defn worm [] 2.138 + (let [model (load-blender-model "Models/worm/worm.blend")] 2.139 + {:body (doto model (body!)) 2.140 + :touch (touch! model) 2.141 + :proprioception (proprioception! model) 2.142 + :muscles (movement! model)})) 2.143 + #+end_src 2.144 + #+end_listing 2.145 + 2.146 ** Embodiment factors action recognition into managable parts 2.147 2.148 + Using empathy, I divide the problem of action recognition into a 2.149 + recognition process expressed in the language of a full compliment 2.150 + of senses, and an imaganitive process that generates full sensory 2.151 + data from partial sensory data. Splitting the action recognition 2.152 + problem in this manner greatly reduces the total amount of work to 2.153 + recognize actions: The imaganitive process is mostly just matching 2.154 + previous experience, and the recognition process gets to use all 2.155 + the senses to directly describe any action. 2.156 + 2.157 ** Action recognition is easy with a full gamut of senses 2.158 2.159 -** Digression: bootstrapping touch using free exploration 2.160 + Embodied representations using multiple senses such as touch, 2.161 + proprioception, and muscle tension turns out be be exceedingly 2.162 + efficient at describing body-centered actions. It is the ``right 2.163 + language for the job''. For example, it takes only around 5 lines 2.164 + of LISP code to describe the action of ``curling'' using embodied 2.165 + primitives. It takes about 8 lines to describe the seemingly 2.166 + complicated action of wiggling. 2.167 + 2.168 + The following action predicates each take a stream of sensory 2.169 + experience, observe however much of it they desire, and decide 2.170 + whether the worm is doing the action they describe. =curled?= 2.171 + relies on proprioception, =resting?= relies on touch, =wiggling?= 2.172 + relies on a fourier analysis of muscle contraction, and 2.173 + =grand-circle?= relies on touch and reuses =curled?= as a gaurd. 2.174 + 2.175 + #+caption: Program for detecting whether the worm is curled. This is the 2.176 + #+caption: simplest action predicate, because it only uses the last frame 2.177 + #+caption: of sensory experience, and only uses proprioceptive data. Even 2.178 + #+caption: this simple predicate, however, is automatically frame 2.179 + #+caption: independent and ignores vermopomorphic differences such as 2.180 + #+caption: worm textures and colors. 2.181 + #+name: curled 2.182 + #+begin_listing clojure 2.183 + #+begin_src clojure 2.184 +(defn curled? 2.185 + "Is the worm curled up?" 2.186 + [experiences] 2.187 + (every? 2.188 + (fn [[_ _ bend]] 2.189 + (> (Math/sin bend) 0.64)) 2.190 + (:proprioception (peek experiences)))) 2.191 + #+end_src 2.192 + #+end_listing 2.193 + 2.194 + #+caption: Program for summarizing the touch information in a patch 2.195 + #+caption: of skin. 2.196 + #+name: touch-summary 2.197 + #+begin_listing clojure 2.198 + #+begin_src clojure 2.199 +(defn contact 2.200 + "Determine how much contact a particular worm segment has with 2.201 + other objects. Returns a value between 0 and 1, where 1 is full 2.202 + contact and 0 is no contact." 2.203 + [touch-region [coords contact :as touch]] 2.204 + (-> (zipmap coords contact) 2.205 + (select-keys touch-region) 2.206 + (vals) 2.207 + (#(map first %)) 2.208 + (average) 2.209 + (* 10) 2.210 + (- 1) 2.211 + (Math/abs))) 2.212 + #+end_src 2.213 + #+end_listing 2.214 + 2.215 + 2.216 + #+caption: Program for detecting whether the worm is at rest. This program 2.217 + #+caption: uses a summary of the tactile information from the underbelly 2.218 + #+caption: of the worm, and is only true if every segment is touching the 2.219 + #+caption: floor. Note that this function contains no references to 2.220 + #+caption: proprioction at all. 2.221 + #+name: resting 2.222 + #+begin_listing clojure 2.223 + #+begin_src clojure 2.224 +(def worm-segment-bottom (rect-region [8 15] [14 22])) 2.225 + 2.226 +(defn resting? 2.227 + "Is the worm resting on the ground?" 2.228 + [experiences] 2.229 + (every? 2.230 + (fn [touch-data] 2.231 + (< 0.9 (contact worm-segment-bottom touch-data))) 2.232 + (:touch (peek experiences)))) 2.233 + #+end_src 2.234 + #+end_listing 2.235 + 2.236 + #+caption: Program for detecting whether the worm is curled up into a 2.237 + #+caption: full circle. Here the embodied approach begins to shine, as 2.238 + #+caption: I am able to both use a previous action predicate (=curled?=) 2.239 + #+caption: as well as the direct tactile experience of the head and tail. 2.240 + #+name: grand-circle 2.241 + #+begin_listing clojure 2.242 + #+begin_src clojure 2.243 +(def worm-segment-bottom-tip (rect-region [15 15] [22 22])) 2.244 + 2.245 +(def worm-segment-top-tip (rect-region [0 15] [7 22])) 2.246 + 2.247 +(defn grand-circle? 2.248 + "Does the worm form a majestic circle (one end touching the other)?" 2.249 + [experiences] 2.250 + (and (curled? experiences) 2.251 + (let [worm-touch (:touch (peek experiences)) 2.252 + tail-touch (worm-touch 0) 2.253 + head-touch (worm-touch 4)] 2.254 + (and (< 0.55 (contact worm-segment-bottom-tip tail-touch)) 2.255 + (< 0.55 (contact worm-segment-top-tip head-touch)))))) 2.256 + #+end_src 2.257 + #+end_listing 2.258 + 2.259 + 2.260 + #+caption: Program for detecting whether the worm has been wiggling for 2.261 + #+caption: the last few frames. It uses a fourier analysis of the muscle 2.262 + #+caption: contractions of the worm's tail to determine wiggling. This is 2.263 + #+caption: signigicant because there is no particular frame that clearly 2.264 + #+caption: indicates that the worm is wiggling --- only when multiple frames 2.265 + #+caption: are analyzed together is the wiggling revealed. Defining 2.266 + #+caption: wiggling this way also gives the worm an opportunity to learn 2.267 + #+caption: and recognize ``frustrated wiggling'', where the worm tries to 2.268 + #+caption: wiggle but can't. Frustrated wiggling is very visually different 2.269 + #+caption: from actual wiggling, but this definition gives it to us for free. 2.270 + #+name: wiggling 2.271 + #+begin_listing clojure 2.272 + #+begin_src clojure 2.273 +(defn fft [nums] 2.274 + (map 2.275 + #(.getReal %) 2.276 + (.transform 2.277 + (FastFourierTransformer. DftNormalization/STANDARD) 2.278 + (double-array nums) TransformType/FORWARD))) 2.279 + 2.280 +(def indexed (partial map-indexed vector)) 2.281 + 2.282 +(defn max-indexed [s] 2.283 + (first (sort-by (comp - second) (indexed s)))) 2.284 + 2.285 +(defn wiggling? 2.286 + "Is the worm wiggling?" 2.287 + [experiences] 2.288 + (let [analysis-interval 0x40] 2.289 + (when (> (count experiences) analysis-interval) 2.290 + (let [a-flex 3 2.291 + a-ex 2 2.292 + muscle-activity 2.293 + (map :muscle (vector:last-n experiences analysis-interval)) 2.294 + base-activity 2.295 + (map #(- (% a-flex) (% a-ex)) muscle-activity)] 2.296 + (= 2 2.297 + (first 2.298 + (max-indexed 2.299 + (map #(Math/abs %) 2.300 + (take 20 (fft base-activity)))))))))) 2.301 + #+end_src 2.302 + #+end_listing 2.303 + 2.304 + With these action predicates, I can now recognize the actions of 2.305 + the worm while it is moving under my control and I have access to 2.306 + all the worm's senses. 2.307 + 2.308 + #+caption: Use the action predicates defined earlier to report on 2.309 + #+caption: what the worm is doing while in simulation. 2.310 + #+name: report-worm-activity 2.311 + #+begin_listing clojure 2.312 + #+begin_src clojure 2.313 +(defn debug-experience 2.314 + [experiences text] 2.315 + (cond 2.316 + (grand-circle? experiences) (.setText text "Grand Circle") 2.317 + (curled? experiences) (.setText text "Curled") 2.318 + (wiggling? experiences) (.setText text "Wiggling") 2.319 + (resting? experiences) (.setText text "Resting"))) 2.320 + #+end_src 2.321 + #+end_listing 2.322 + 2.323 + #+caption: Using =debug-experience=, the body-centered predicates 2.324 + #+caption: work together to classify the behaviour of the worm. 2.325 + #+caption: while under manual motor control. 2.326 + #+name: basic-worm-view 2.327 + #+ATTR_LaTeX: :width 10cm 2.328 + [[./images/worm-identify-init.png]] 2.329 + 2.330 + These action predicates satisfy the recognition requirement of an 2.331 + empathic recognition system. There is a lot of power in the 2.332 + simplicity of the action predicates. They describe their actions 2.333 + without getting confused in visual details of the worm. Each one is 2.334 + frame independent, but more than that, they are each indepent of 2.335 + irrelevant visual details of the worm and the environment. They 2.336 + will work regardless of whether the worm is a different color or 2.337 + hevaily textured, or of the environment has strange lighting. 2.338 + 2.339 + The trick now is to make the action predicates work even when the 2.340 + sensory data on which they depend is absent. If I can do that, then 2.341 + I will have gained much, 2.342 2.343 ** \Phi-space describes the worm's experiences 2.344 + 2.345 + As a first step towards building empathy, I need to gather all of 2.346 + the worm's experiences during free play. I use a simple vector to 2.347 + store all the experiences. 2.348 + 2.349 + #+caption: Program to gather the worm's experiences into a vector for 2.350 + #+caption: further processing. The =motor-control-program= line uses 2.351 + #+caption: a motor control script that causes the worm to execute a series 2.352 + #+caption: of ``exercices'' that include all the action predicates. 2.353 + #+name: generate-phi-space 2.354 + #+begin_listing clojure 2.355 + #+begin_src clojure 2.356 +(defn generate-phi-space [] 2.357 + (let [experiences (atom [])] 2.358 + (run-world 2.359 + (apply-map 2.360 + worm-world 2.361 + (merge 2.362 + (worm-world-defaults) 2.363 + {:end-frame 700 2.364 + :motor-control 2.365 + (motor-control-program worm-muscle-labels do-all-the-things) 2.366 + :experiences experiences}))) 2.367 + @experiences)) 2.368 + #+end_src 2.369 + #+end_listing 2.370 + 2.371 + Each element of the experience vector exists in the vast space of 2.372 + all possible worm-experiences. Most of this vast space is actually 2.373 + unreachable due to physical constraints of the worm's body. For 2.374 + example, the worm's segments are connected by hinge joints that put 2.375 + a practical limit on the worm's degrees of freedom. Also, the worm 2.376 + can not be bent into a circle so that its ends are touching and at 2.377 + the same time not also experience the sensation of touching itself. 2.378 + 2.379 + As the worm moves around during free play and the vector grows 2.380 + larger, the vector begins to define a subspace which is all the 2.381 + practical experiences the worm can experience during normal 2.382 + operation, which I call \Phi-space, short for physical-space. The 2.383 + vector defines a path through \Phi-space. This path has interesting 2.384 + properties that all derive from embodiment. The proprioceptive 2.385 + components are completely smooth, because in order for the worm to 2.386 + move from one position to another, it must pass through the 2.387 + intermediate positions. The path invariably forms loops as actions 2.388 + are repeated. Finally and most importantly, proprioception actually 2.389 + gives very strong inference about the other senses. For example, 2.390 + when the worm is flat, you can infer that it is touching the ground 2.391 + and that its muscles are not active, because if the muscles were 2.392 + active, the worm would be moving and would not be perfectly flat. 2.393 + In order to stay flat, the worm has to be touching the ground, or 2.394 + it would again be moving out of the flat position due to gravity. 2.395 + If the worm is positioned in such a way that it interacts with 2.396 + itself, then it is very likely to be feeling the same tactile 2.397 + feelings as the last time it was in that position, because it has 2.398 + the same body as then. If you observe multiple frames of 2.399 + proprioceptive data, then you can become increasingly confident 2.400 + about the exact activations of the worm's muscles, because it 2.401 + generally takes a unique combination of muscle contractions to 2.402 + transform the worm's body along a specific path through \Phi-space. 2.403 + 2.404 + There is a simple way of taking \Phi-space and the total ordering 2.405 + provided by an experience vector and reliably infering the rest of 2.406 + the senses. 2.407 2.408 ** Empathy is the process of tracing though \Phi-space 2.409 + 2.410 + 2.411 + 2.412 +(defn bin [digits] 2.413 + (fn [angles] 2.414 + (->> angles 2.415 + (flatten) 2.416 + (map (juxt #(Math/sin %) #(Math/cos %))) 2.417 + (flatten) 2.418 + (mapv #(Math/round (* % (Math/pow 10 (dec digits)))))))) 2.419 + 2.420 +(defn gen-phi-scan 2.421 +"Nearest-neighbors with spatial binning. Only returns a result if 2.422 + the propriceptive data is within 10% of a previously recorded 2.423 + result in all dimensions." 2.424 + 2.425 +[phi-space] 2.426 + (let [bin-keys (map bin [3 2 1]) 2.427 + bin-maps 2.428 + (map (fn [bin-key] 2.429 + (group-by 2.430 + (comp bin-key :proprioception phi-space) 2.431 + (range (count phi-space)))) bin-keys) 2.432 + lookups (map (fn [bin-key bin-map] 2.433 + (fn [proprio] (bin-map (bin-key proprio)))) 2.434 + bin-keys bin-maps)] 2.435 + (fn lookup [proprio-data] 2.436 + (set (some #(% proprio-data) lookups))))) 2.437 + 2.438 + 2.439 +(defn longest-thread 2.440 + "Find the longest thread from phi-index-sets. The index sets should 2.441 + be ordered from most recent to least recent." 2.442 + [phi-index-sets] 2.443 + (loop [result '() 2.444 + [thread-bases & remaining :as phi-index-sets] phi-index-sets] 2.445 + (if (empty? phi-index-sets) 2.446 + (vec result) 2.447 + (let [threads 2.448 + (for [thread-base thread-bases] 2.449 + (loop [thread (list thread-base) 2.450 + remaining remaining] 2.451 + (let [next-index (dec (first thread))] 2.452 + (cond (empty? remaining) thread 2.453 + (contains? (first remaining) next-index) 2.454 + (recur 2.455 + (cons next-index thread) (rest remaining)) 2.456 + :else thread)))) 2.457 + longest-thread 2.458 + (reduce (fn [thread-a thread-b] 2.459 + (if (> (count thread-a) (count thread-b)) 2.460 + thread-a thread-b)) 2.461 + '(nil) 2.462 + threads)] 2.463 + (recur (concat longest-thread result) 2.464 + (drop (count longest-thread) phi-index-sets)))))) 2.465 + 2.466 +There is one final piece, which is to replace missing sensory data 2.467 +with a best-guess estimate. While I could fill in missing data by 2.468 +using a gradient over the closest known sensory data points, averages 2.469 +can be misleading. It is certainly possible to create an impossible 2.470 +sensory state by averaging two possible sensory states. Therefore, I 2.471 +simply replicate the most recent sensory experience to fill in the 2.472 +gaps. 2.473 + 2.474 + #+caption: Fill in blanks in sensory experience by replicating the most 2.475 + #+caption: recent experience. 2.476 + #+name: infer-nils 2.477 + #+begin_listing clojure 2.478 + #+begin_src clojure 2.479 +(defn infer-nils 2.480 + "Replace nils with the next available non-nil element in the 2.481 + sequence, or barring that, 0." 2.482 + [s] 2.483 + (loop [i (dec (count s)) 2.484 + v (transient s)] 2.485 + (if (zero? i) (persistent! v) 2.486 + (if-let [cur (v i)] 2.487 + (if (get v (dec i) 0) 2.488 + (recur (dec i) v) 2.489 + (recur (dec i) (assoc! v (dec i) cur))) 2.490 + (recur i (assoc! v i 0)))))) 2.491 + #+end_src 2.492 + #+end_listing 2.493 + 2.494 + 2.495 + 2.496 + 2.497 2.498 ** Efficient action recognition with =EMPATH= 2.499 2.500 +** Digression: bootstrapping touch using free exploration 2.501 + 2.502 * Contributions 2.503 - - Built =CORTEX=, a comprehensive platform for embodied AI 2.504 - experiments. Has many new features lacking in other systems, such 2.505 - as sound. Easy to model/create new creatures. 2.506 - - created a novel concept for action recognition by using artificial 2.507 - imagination. 2.508 - 2.509 -In the second half of the thesis I develop a computational model of 2.510 -empathy, using =CORTEX= as a base. Empathy in this context is the 2.511 -ability to observe another creature and infer what sorts of sensations 2.512 -that creature is feeling. My empathy algorithm involves multiple 2.513 -phases. First is free-play, where the creature moves around and gains 2.514 -sensory experience. From this experience I construct a representation 2.515 -of the creature's sensory state space, which I call \Phi-space. Using 2.516 -\Phi-space, I construct an efficient function for enriching the 2.517 -limited data that comes from observing another creature with a full 2.518 -compliment of imagined sensory data based on previous experience. I 2.519 -can then use the imagined sensory data to recognize what the observed 2.520 -creature is doing and feeling, using straightforward embodied action 2.521 -predicates. This is all demonstrated with using a simple worm-like 2.522 -creature, and recognizing worm-actions based on limited data. 2.523 - 2.524 -Embodied representation using multiple senses such as touch, 2.525 -proprioception, and muscle tension turns out be be exceedingly 2.526 -efficient at describing body-centered actions. It is the ``right 2.527 -language for the job''. For example, it takes only around 5 lines of 2.528 -LISP code to describe the action of ``curling'' using embodied 2.529 -primitives. It takes about 8 lines to describe the seemingly 2.530 -complicated action of wiggling. 2.531 - 2.532 - 2.533 - 2.534 -* COMMENT names for cortex 2.535 - - bioland 2.536 2.537 2.538
3.1 Binary file thesis/images/basic-worm-view.png has changed
4.1 Binary file thesis/images/full-hand.png has changed
5.1 Binary file thesis/images/worm-identify-init.png has changed
6.1 --- a/thesis/rlm-cortex-meng.tex Tue Mar 25 22:54:41 2014 -0400 6.2 +++ b/thesis/rlm-cortex-meng.tex Wed Mar 26 02:42:01 2014 -0400 6.3 @@ -43,7 +43,7 @@ 6.4 \usepackage{hyperref} 6.5 \usepackage{libertine} 6.6 \usepackage{inconsolata} 6.7 - 6.8 +\usepackage{rotating} 6.9 6.10 \usepackage[backend=bibtex,style=alphabetic]{biblatex} 6.11 \addbibresource{cortex.bib}