Mercurial > cortex
changeset 266:bee5145ce463
Merged Robert's proprioception with Dylan's vision.
author | Dylan Holmes <ocsenave@gmail.com> |
---|---|
date | Mon, 13 Feb 2012 21:53:54 -0600 (2012-02-14) |
parents | e57d8c52f12f (current diff) 2fdcbe8185b1 (diff) |
children | 6446e964810f |
files | |
diffstat | 10 files changed, 642 insertions(+), 192 deletions(-) [+] |
line wrap: on
line diff
1.1 Binary file assets/Models/test-creature/basic-muscle.png has changed
2.1 Binary file assets/Models/test-creature/worm.blend has changed
3.1 Binary file assets/Models/test-touch/touch-cube.blend has changed
4.1 Binary file images/basic-muscle.png has changed
5.1 Binary file images/worm-with-muscle.png has changed
6.1 --- a/org/hearing.org Mon Feb 13 21:53:28 2012 -0600 6.2 +++ b/org/hearing.org Mon Feb 13 21:53:54 2012 -0600 6.3 @@ -950,7 +950,6 @@ 6.4 #+end_src 6.5 6.6 * Testing Hearing 6.7 - 6.8 ** Advanced Java Example 6.9 6.10 I wrote a test case in Java that demonstrates the use of the Java 6.11 @@ -1083,7 +1082,6 @@ 6.12 sound it hears beomes fainter. This shows the 3D localization of 6.13 sound in this world.</p> 6.14 </div> 6.15 - 6.16 #+end_html 6.17 6.18 *** Creating the Ear Video 6.19 @@ -1169,7 +1167,6 @@ 6.20 #+html: <ul> <li> <a href="../org/hearing.org">This org file</a> </li> </ul> 6.21 - [[http://hg.bortreb.com ][source-repository]] 6.22 6.23 - 6.24 * Next 6.25 The worm can see and hear, but it can't feel the world or 6.26 itself. Next post, I'll give the worm a [[./touch.org][sense of touch]].
7.1 --- a/org/ideas.org Mon Feb 13 21:53:28 2012 -0600 7.2 +++ b/org/ideas.org Mon Feb 13 21:53:54 2012 -0600 7.3 @@ -100,6 +100,7 @@ 7.4 - [ ] show sensor maps in HUD display? -- 4 days 7.5 - [ ] show sensor maps in AWT display? -- 2 days 7.6 - [ ] upgrade to clojure 1.3, replace all defvars with new def 7.7 + - [ ] common video creation code. 7.8 7.9 7.10
8.1 --- a/org/movement.org Mon Feb 13 21:53:28 2012 -0600 8.2 +++ b/org/movement.org Mon Feb 13 21:53:54 2012 -0600 8.3 @@ -1,4 +1,4 @@ 8.4 -#+title: Movement! 8.5 +#+title: Simulated Muscles 8.6 #+author: Robert McIntyre 8.7 #+email: rlm@mit.edu 8.8 #+description: muscles for a simulated creature 8.9 @@ -7,48 +7,61 @@ 8.10 #+INCLUDE: ../../aurellem/org/level-0.org 8.11 8.12 8.13 +* Muscles 8.14 + 8.15 Surprisingly enough, terristerial creatures only move by using torque 8.16 applied about their joints. There's not a single straight line of 8.17 force in the human body at all! (A straight line of force would 8.18 -correspond to some sort of jet or rocket propulseion.) 8.19 +correspond to some sort of jet or rocket propulsion.) 8.20 8.21 +*(next paragraph is from memory and needs to be checked!)* 8.22 8.23 -Here's how motor-control/ proprioception will work: Each muscle is 8.24 -defined by a 1-D array of numbers (the "motor pool") each of which 8.25 -represent muscle fibers. A muscle also has a scalar :strength factor 8.26 -which determines how strong the muscle as a whole is. The effector 8.27 -function for a muscle takes a number < (count motor-pool) and that 8.28 -number is said to "activate" all the muscle fibers whose index is 8.29 -lower than the number. Each fiber will apply force in proportion to 8.30 -its value in the array. Lower values cause less force. The lower 8.31 -values can be put at the "beginning" of the 1-D array to simulate the 8.32 -layout of actual human muscles, which are capable of more percise 8.33 -movements when exerting less force. 8.34 +In humans, muscles are composed of millions of sarcomeres, which can 8.35 +contract to exert force. A single motor neuron might control 100-1,000 8.36 +sarcomeres. When the motor neuron is engaged by the brain, it 8.37 +activates all of the sarcomeres to which it is attached. Some motor 8.38 +neurons command many sarcomeres, and some command only a few. The 8.39 +spinal cord generally engages the motor neurons which control few 8.40 +sarcomeres before the motor neurons which control many sarcomeres. 8.41 +This recruitment stragety allows for percise movements at low 8.42 +strength. The collection of all motor neurons that control a muscle is 8.43 +called the motor pool. The brain essentially says "activate 30% of the 8.44 +motor pool" and the spinal cord recruits motor neurons untill 30% are 8.45 +activated. Since the distribution of power among motor neurons is 8.46 +unequal and recruitment goes from weakest to strongest, 30% of the 8.47 +motor pool might be 5% of the strength of the muscle. 8.48 8.49 -#+name: movement 8.50 +My simulated muscles follow a similiar design: Each muscle is defined 8.51 +by a 1-D array of numbers (the "motor pool"). Each number represents a 8.52 +motor neuron which controlls a number of sarcomeres equal to the 8.53 +number. A muscle also has a scalar :strength factor which determines 8.54 +the total force the muscle can exert when all motor neurons are 8.55 +activated. The effector function for a muscle takes a number to index 8.56 +into the motor pool, and that number "activates" all the motor neurons 8.57 +whose index is lower or equal to the number. Each motor-neuron will 8.58 +apply force in proportion to its value in the array. Lower values 8.59 +cause less force. The lower values can be put at the "beginning" of 8.60 +the 1-D array to simulate the layout of actual human muscles, which 8.61 +are capable of more percise movements when exerting less force. Or, 8.62 +the motor pool can simulate more exoitic recruitment strageties which 8.63 +do not correspond to human muscles. 8.64 + 8.65 +This 1D array is defined in an image file for ease of 8.66 +creation/visualization. Here is an example muscle profile image. 8.67 + 8.68 +#+caption: A muscle profile image that describes the strengths of each motor neuron in a muscle. White is weakest and dark red is strongest. This particular pattern has weaker motor neurons at the beginning, just like human muscle. 8.69 +[[../images/basic-muscle.png]] 8.70 + 8.71 +* Blender Meta-data 8.72 + 8.73 +In blender, each muscle is an empty node whose top level parent is 8.74 +named "muscles", just like eyes, ears, and joints. 8.75 + 8.76 +These functions define the expected meta-data for a muscle node. 8.77 + 8.78 +#+name: muscle-meta-data 8.79 #+begin_src clojure 8.80 -(ns cortex.movement 8.81 - "Give simulated creatures defined in special blender files the power 8.82 - to move around in a simulated environment." 8.83 - {:author "Robert McIntyre"} 8.84 - (:use (cortex world util sense body)) 8.85 - (:use clojure.contrib.def) 8.86 - (:import java.awt.image.BufferedImage) 8.87 - (:import com.jme3.scene.Node) 8.88 - (:import com.jme3.math.Vector3f) 8.89 - (:import com.jme3.bullet.control.RigidBodyControl)) 8.90 - 8.91 -(defn muscle-profile 8.92 - "Return a vector where each entry is the strength of the \"motor 8.93 - pool\" at that part in the muscle." 8.94 - [#^BufferedImage profile] 8.95 - (vec 8.96 - (let [width (.getWidth profile)] 8.97 - (for [x (range width)] 8.98 - (- 255 8.99 - (bit-and 8.100 - 0x0000FF 8.101 - (.getRGB profile x 0))))))) 8.102 +(in-ns 'cortex.movement) 8.103 8.104 (defvar 8.105 ^{:arglists '([creature])} 8.106 @@ -56,30 +69,64 @@ 8.107 (sense-nodes "muscles") 8.108 "Return the children of the creature's \"muscles\" node.") 8.109 8.110 -(defn movement-fn 8.111 +(defn muscle-profile-image 8.112 + "Get the muscle-profile image from the node's blender meta-data." 8.113 + [#^Node muscle] 8.114 + (if-let [image (meta-data muscle "muscle")] 8.115 + (load-image image))) 8.116 + 8.117 +(defn muscle-strength 8.118 + "Return the strength of this muscle, or 1 if it is not defined." 8.119 + [#^Node muscle] 8.120 + (if-let [strength (meta-data muscle "strength")] 8.121 + strength 1)) 8.122 + 8.123 +(defn motor-pool 8.124 + "Return a vector where each entry is the strength of the \"motor 8.125 + neuron\" at that part in the muscle." 8.126 + [#^Node muscle] 8.127 + (let [profile (muscle-profile-image muscle)] 8.128 + (vec 8.129 + (let [width (.getWidth profile)] 8.130 + (for [x (range width)] 8.131 + (- 255 8.132 + (bit-and 8.133 + 0x0000FF 8.134 + (.getRGB profile x 0)))))))) 8.135 +#+end_src 8.136 + 8.137 +Of note here is =(motor-pool)= which interprets the muscle-profile 8.138 +image in a way that allows me to use gradients between white and red, 8.139 +instead of shades of gray as I've been using for all the other 8.140 +senses. This is purely an aesthetic touch. 8.141 + 8.142 +* Creating Muscles 8.143 +#+name: muscle-kernel 8.144 +#+begin_src clojure 8.145 +(in-ns 'cortex.movement) 8.146 + 8.147 +(defn movement-kernel 8.148 "Returns a function which when called with a integer value inside a 8.149 running simulation will cause movement in the creature according 8.150 to the muscle's position and strength profile. Each function 8.151 returns the amount of force applied / max force." 8.152 - [#^Node parts #^Node muscle] 8.153 - (let [target (closest-node parts muscle) 8.154 + [#^Node creature #^Node muscle] 8.155 + (let [target (closest-node creature muscle) 8.156 axis 8.157 (.mult (.getWorldRotation muscle) Vector3f/UNIT_Y) 8.158 - strength (meta-data muscle "strength") 8.159 - image-name (read-string (meta-data muscle "muscle")) 8.160 - image (load-image image-name) 8.161 - fibers (muscle-profile image) 8.162 - fiber-integral (reductions + fibers) 8.163 + strength (muscle-strength muscle) 8.164 + 8.165 + pool (motor-pool muscle) 8.166 + pool-integral (reductions + pool) 8.167 force-index 8.168 - (vec (map #(float (* strength (/ % (last fiber-integral)))) 8.169 - fiber-integral)) 8.170 + (vec (map #(float (* strength (/ % (last pool-integral)))) 8.171 + pool-integral)) 8.172 control (.getControl target RigidBodyControl)] 8.173 (fn [n] 8.174 - (let [pool-index (max 0 (min n (dec (count fibers)))) 8.175 + (let [pool-index (max 0 (min n (dec (count pool)))) 8.176 force (force-index pool-index)] 8.177 (.applyTorque control (.mult axis force)) 8.178 (float (/ force strength)))))) 8.179 - 8.180 8.181 (defn movement! 8.182 "Endow the creature with the power of movement. Returns a sequence 8.183 @@ -87,8 +134,24 @@ 8.184 activate their corresponding muscle." 8.185 [#^Node creature] 8.186 (for [muscle (muscles creature)] 8.187 - (movement-fn creature muscle))) 8.188 + (movement-kernel creature muscle))) 8.189 +#+end_src 8.190 8.191 +=(movement-kernel)= creates a function that will move the nearest 8.192 +physical object to the muscle node. The muscle exerts a rotational 8.193 +force dependant on it's orientation to the object in the blender 8.194 +file. The function returned by =(movement-kernel)= is also a sense 8.195 +function: it returns the percent of the total muscle strength that is 8.196 +currently being employed. This is analogous to muscle tension in 8.197 +humans and completes the sense of proprioception begun in the last 8.198 +post. 8.199 + 8.200 +* Visualizing Muscle Tension 8.201 +Muscle exertion is a percent of a total, so the visulazation is just a 8.202 +simple percent bar. 8.203 + 8.204 +#+name: visualization 8.205 +#+begin_src clojure 8.206 (defn movement-display-kernel 8.207 "Display muscle exertion data as a bar filling up with red." 8.208 [exertion] 8.209 @@ -108,14 +171,166 @@ 8.210 displays each element of the list to the screen." 8.211 [] 8.212 (view-sense movement-display-kernel)) 8.213 - 8.214 #+end_src 8.215 8.216 +* Adding Touch to the Worm 8.217 8.218 +#+begin_src clojure 8.219 +(defn test-movement 8.220 + ([] (test-movement false)) 8.221 + ([record?] 8.222 + (let [creature (doto (worm) (body!)) 8.223 8.224 + muscle-exertion (atom 0) 8.225 + muscles (movement! creature) 8.226 + muscle-display (view-movement)] 8.227 + (.setMass 8.228 + (.getControl (.getChild creature "worm-11") RigidBodyControl) 8.229 + (float 0)) 8.230 + (world 8.231 + (nodify [creature (floor)]) 8.232 + (merge standard-debug-controls 8.233 + {"key-h" 8.234 + (fn [_ value] 8.235 + (if value 8.236 + (swap! muscle-exertion (partial + 20)))) 8.237 + "key-n" 8.238 + (fn [_ value] 8.239 + (if value 8.240 + (swap! muscle-exertion (fn [v] (- v 20)))))}) 8.241 + (fn [world] 8.242 + (if record? 8.243 + (Capture/captureVideo 8.244 + world 8.245 + (File. "/home/r/proj/cortex/render/worm-muscles/main-view"))) 8.246 + (light-up-everything world) 8.247 + (enable-debug world) 8.248 + (.setTimer world (RatchetTimer. 60)) 8.249 + (set-gravity world (Vector3f. 0 0 0)) 8.250 + (.setLocation (.getCamera world) 8.251 + (Vector3f. -4.912815, 2.004171, 0.15710819)) 8.252 + (.setRotation (.getCamera world) 8.253 + (Quaternion. 0.13828252, 0.65516764, 8.254 + -0.12370994, 0.7323449)) 8.255 8.256 + (comment 8.257 + (com.aurellem.capture.Capture/captureVideo 8.258 + world (file-str "/home/r/proj/ai-videos/hand")))) 8.259 + (fn [world tpf] 8.260 + (muscle-display 8.261 + (map #(% @muscle-exertion) muscles) 8.262 + (if record? 8.263 + (File. "/home/r/proj/cortex/render/worm-muscles/muscles")))))))) 8.264 +#+end_src 8.265 + 8.266 +* Video Demonstration 8.267 + 8.268 +#+begin_html 8.269 +<div class="figure"> 8.270 +<center> 8.271 +<video controls="controls" width="550"> 8.272 + <source src="../video/worm-muscles.ogg" type="video/ogg" 8.273 + preload="none" poster="../images/aurellem-1280x480.png" /> 8.274 +</video> 8.275 +</center> 8.276 +<p>The worm is now able to move. The bar in the lower right displays 8.277 + the power output of the muscle . Each jump causes 20 more motor neurons to 8.278 + be recruited. Notice that the power output increases non-linearly 8.279 + with motror neuron recruitement, similiar to a human muscle.</p> 8.280 +</div> 8.281 +#+end_html 8.282 + 8.283 + 8.284 +** Making the Worm Muscles Video 8.285 +#+name: magick7 8.286 +#+begin_src clojure 8.287 +(ns cortex.video.magick7 8.288 + (:import java.io.File) 8.289 + (:use clojure.contrib.shell-out)) 8.290 + 8.291 +(defn images [path] 8.292 + (sort (rest (file-seq (File. path))))) 8.293 + 8.294 +(def base "/home/r/proj/cortex/render/worm-muscles/") 8.295 + 8.296 +(defn pics [file] 8.297 + (images (str base file))) 8.298 + 8.299 +(defn combine-images [] 8.300 + (let [main-view (pics "main-view") 8.301 + muscles (pics "muscles/0") 8.302 + targets (map 8.303 + #(File. (str base "out/" (format "%07d.png" %))) 8.304 + (range 0 (count main-view)))] 8.305 + (dorun 8.306 + (pmap 8.307 + (comp 8.308 + (fn [[ main-view muscles target]] 8.309 + (println target) 8.310 + (sh "convert" 8.311 + main-view 8.312 + muscles "-geometry" "+320+440" "-composite" 8.313 + target)) 8.314 + (fn [& args] (map #(.getCanonicalPath %) args))) 8.315 + main-view muscles targets)))) 8.316 +#+end_src 8.317 + 8.318 +#+begin_src sh :results silent 8.319 +cd ~/proj/cortex/render/worm-muscles 8.320 +ffmpeg -r 60 -i out/%07d.png -b:v 9000k -c:v libtheora worm-muscles.ogg 8.321 +#+end_src 8.322 + 8.323 +* Headers 8.324 +#+name: muscle-header 8.325 +#+begin_src clojure 8.326 +(ns cortex.movement 8.327 + "Give simulated creatures defined in special blender files the power 8.328 + to move around in a simulated environment." 8.329 + {:author "Robert McIntyre"} 8.330 + (:use (cortex world util sense body)) 8.331 + (:use clojure.contrib.def) 8.332 + (:import java.awt.image.BufferedImage) 8.333 + (:import com.jme3.scene.Node) 8.334 + (:import com.jme3.math.Vector3f) 8.335 + (:import com.jme3.bullet.control.RigidBodyControl)) 8.336 +#+end_src 8.337 + 8.338 +#+name: test-header 8.339 +#+begin_src clojure 8.340 +(ns cortex.test.movement 8.341 + (:use (cortex world util sense body movement)) 8.342 + (:use cortex.test.body) 8.343 + (:use clojure.contrib.def) 8.344 + (:import java.io.File) 8.345 + (:import java.awt.image.BufferedImage) 8.346 + (:import com.jme3.scene.Node) 8.347 + (:import com.jme3.math.Vector3f) 8.348 + (:import (com.aurellem.capture Capture RatchetTimer)) 8.349 + (:import com.jme3.bullet.control.RigidBodyControl)) 8.350 + 8.351 +(cortex.import/mega-import-jme3) 8.352 +#+end_src 8.353 + 8.354 +* Source Listing 8.355 + - [[../src/cortex/movement.clj][cortex.movement]] 8.356 + - [[../src/cortex/test/movement.clj][cortex.test.movement]] 8.357 + - [[../src/cortex/video/magick7.clj][cortex.video.magick7]] 8.358 +#+html: <ul> <li> <a href="../org/movement.org">This org file</a> </li> </ul> 8.359 + - [[http://hg.bortreb.com ][source-repository]] 8.360 8.361 * COMMENT code generation 8.362 #+begin_src clojure :tangle ../src/cortex/movement.clj 8.363 -<<movement>> 8.364 +<<muscle-header>> 8.365 +<<muscle-meta-data>> 8.366 +<<muscle-kernel>> 8.367 +<<visualization>> 8.368 #+end_src 8.369 + 8.370 +#+begin_src clojure :tangle ../src/cortex/test/movement.clj 8.371 +<<test-header>> 8.372 +<<test-movement>> 8.373 +#+end_src 8.374 + 8.375 +#+begin_src clojure :tangle ../src/cortex/video/magick7.clj 8.376 +<<magick7>> 8.377 +#+end_src
9.1 --- a/org/proprioception.org Mon Feb 13 21:53:28 2012 -0600 9.2 +++ b/org/proprioception.org Mon Feb 13 21:53:54 2012 -0600 9.3 @@ -6,36 +6,89 @@ 9.4 #+SETUPFILE: ../../aurellem/org/setup.org 9.5 #+INCLUDE: ../../aurellem/org/level-0.org 9.6 9.7 -#+name: proprioception 9.8 +* Proprioception 9.9 + 9.10 +Close your eyes, and touch your nose with your right index finger. How 9.11 +did you do it? You could not see your hand, and neither your hand nor 9.12 +your nose could use the sense of touch to guide the path of your hand. 9.13 +There are no sound cues, and Taste and Smell certainly don't provide 9.14 +any help. You know where your hand is without your other senses 9.15 +because of Proprioception. 9.16 + 9.17 +Humans can sometimes loose this sense through viral infections or 9.18 +damage to the spinal cord or brain, and when they do, they loose the 9.19 +ability to control their own bodies without looking directly at the 9.20 +parts they want to move. In [[http://en.wikipedia.org/wiki/The_Man_Who_Mistook_His_Wife_for_a_Hat][The Man Who Mistook His Wife for a Hat]], 9.21 +a woman named Christina looses this sense and has to learn how to move 9.22 +by carefully watching her arms and legs. She describes proprioception 9.23 +as the "eyes of the body, the way the body sees itself". 9.24 + 9.25 +Proprioception in humans is mediated by [[http://en.wikipedia.org/wiki/Articular_capsule][joint capsules]], [[http://en.wikipedia.org/wiki/Muscle_spindle][muscle 9.26 +spindles]], and the [[http://en.wikipedia.org/wiki/Golgi_tendon_organ][Golgi tendon organs]]. These measure the relative 9.27 +positions of each pody part by monitoring muscle strain and length. 9.28 + 9.29 +It's clear that this is a vital sense for fulid, graceful 9.30 +movement. It's also particurally easy to implement in jMonkeyEngine. 9.31 + 9.32 +My simulated proprioception calculates the relative angles of each 9.33 +joint from the rest position defined in the blender file. This 9.34 +simulates the muscle-spindles and joint capsules. I will deal with 9.35 +Golgi tendon organs, which calculate muscle strain, in the [[./movement.org][next post]]. 9.36 + 9.37 +* Helper Functions 9.38 + 9.39 +=(absolute-angle)= calculates the angle between two vectors, relative to a 9.40 +third axis vector. This angle is the number of radians you have to 9.41 +move counterclockwise around the axis vector to get from the first to 9.42 +the second vector. It is not commutative like a normal dot-product 9.43 +angle is. 9.44 + 9.45 +#+name: helpers 9.46 #+begin_src clojure 9.47 -(ns cortex.proprioception 9.48 - "Simulate the sense of proprioception (ability to detect the 9.49 - relative positions of body parts with repsect to other body parts) 9.50 - in jMonkeyEngine3. Reads specially prepared blender files to 9.51 - automatically generate proprioceptive senses." 9.52 - (:use (cortex world util sense body)) 9.53 - (:use clojure.contrib.def) 9.54 - (:import com.jme3.scene.Node) 9.55 - (:import java.awt.image.BufferedImage) 9.56 - (:import (com.jme3.math Vector3f Quaternion))) 9.57 +(in-ns 'cortex.proprioception) 9.58 9.59 (defn right-handed? 9.60 "true iff the three vectors form a right handed coordinate 9.61 - system. The three vectors do not have to be normalized or 9.62 - orthogonal." 9.63 + system. The three vectors do not have to be normalized or 9.64 + orthogonal." 9.65 [vec1 vec2 vec3] 9.66 (< 0 (.dot (.cross vec1 vec2) vec3))) 9.67 9.68 (defn absolute-angle 9.69 - "The angle between 'vec1 and 'vec2. Positive if the angle to get 9.70 - from 'vec1 to 'vec2 is counterclockwise around 'axis, and negative 9.71 - otherwise." 9.72 + "The angle between 'vec1 and 'vec2 around 'axis. In the range 9.73 + [0 (* 2 Math/PI)]." 9.74 [vec1 vec2 axis] 9.75 (let [angle (.angleBetween vec1 vec2)] 9.76 (if (right-handed? vec1 vec2 axis) 9.77 angle (- (* 2 Math/PI) angle)))) 9.78 +#+end_src 9.79 9.80 -(defn proprioception-fn 9.81 +#+begin_src clojure :exports both 9.82 +(in-ns 'cortex.proprioception) 9.83 +(absolute-angle Vector3f/UNIT_X Vector3f/UNIT_Y Vector3f/UNIT_Z) 9.84 +#+end_src 9.85 + 9.86 +#+results: 9.87 +: 1.5707964 9.88 + 9.89 +#+begin_src clojure :exports both 9.90 +(in-ns 'cortex.proprioception) 9.91 +(absolute-angle 9.92 + Vector3f/UNIT_X (.mult Vector3f/UNIT_Y (float -1)) Vector3f/UNIT_Z) 9.93 +#+end_src 9.94 + 9.95 +#+results: 9.96 +: 4.7123889366733 9.97 + 9.98 +* Proprioception Kernel 9.99 + 9.100 +Given a joint, =(proprioception-kernel)= produces a function that 9.101 +calculates the euler angles between the the objects the joint 9.102 +connects. 9.103 + 9.104 +#+name: proprioception 9.105 +#+begin_src clojure 9.106 +(defn proprioception-kernel 9.107 "Returns a function which returns proprioceptive sensory data when 9.108 called inside a running simulation." 9.109 [#^Node parts #^Node joint] 9.110 @@ -44,12 +97,6 @@ 9.111 x0 (.mult joint-rot Vector3f/UNIT_X) 9.112 y0 (.mult joint-rot Vector3f/UNIT_Y) 9.113 z0 (.mult joint-rot Vector3f/UNIT_Z)] 9.114 - (println-repl "x:" x0) 9.115 - (println-repl "y:" y0) 9.116 - (println-repl "z:" z0) 9.117 - (println-repl "init-a:" (.getWorldRotation obj-a)) 9.118 - (println-repl "init-b:" (.getWorldRotation obj-b)) 9.119 - 9.120 (fn [] 9.121 (let [rot-a (.clone (.getWorldRotation obj-a)) 9.122 rot-b (.clone (.getWorldRotation obj-b)) 9.123 @@ -80,18 +127,33 @@ 9.124 that joint." 9.125 [#^Node creature] 9.126 ;; extract the body's joints 9.127 - (let [senses (map (partial proprioception-fn creature) 9.128 + (let [senses (map (partial proprioception-kernel creature) 9.129 (joints creature))] 9.130 (fn [] 9.131 (map #(%) senses)))) 9.132 +#+end_src 9.133 9.134 9.135 +=(proprioception!)= maps =(proprioception-kernel)= across all the 9.136 +joints of the creature. It uses the same list of joints that 9.137 +=(cortex.body/joints)= uses. 9.138 + 9.139 +* Visualizing Proprioception 9.140 + 9.141 +Proprioception has the lowest bandwidth of all the senses so far, and 9.142 +it doesn't lend itself as readily to visual representation like 9.143 +vision, hearing, or touch. This visualization code creates a "guage" 9.144 +to view each of the three relative angles along a circle. 9.145 + 9.146 +#+name: visualize 9.147 +#+begin_src clojure 9.148 +(in-ns 'cortex.proprioception) 9.149 + 9.150 (defn draw-sprite [image sprite x y color ] 9.151 (dorun 9.152 (for [[u v] sprite] 9.153 (.setRGB image (+ u x) (+ v y) color)))) 9.154 9.155 - 9.156 (defn view-angle 9.157 "create a debug view of an angle" 9.158 [color] 9.159 @@ -109,7 +171,6 @@ 9.160 (reset! previous position)) 9.161 image)))) 9.162 9.163 - 9.164 (defn proprioception-display-kernel 9.165 "Display proprioception angles in a BufferedImage" 9.166 [[h p r]] 9.167 @@ -143,14 +204,16 @@ 9.168 display each element of the list to the screen as an image." 9.169 [] 9.170 (view-sense proprioception-display-kernel)) 9.171 - 9.172 - 9.173 - 9.174 #+end_src 9.175 9.176 -#+name: test-body 9.177 +* Proprioception Test 9.178 +This test does not use the worm, but instead uses two bars, bound 9.179 +together by a point2point joint. One bar is fixed, and I control the 9.180 +other bar from the keyboard. 9.181 + 9.182 +#+name: test-proprioception 9.183 #+begin_src clojure 9.184 - 9.185 +(in-ns 'cortex.test.proprioception) 9.186 9.187 (defn test-proprioception 9.188 "Testing proprioception: 9.189 @@ -159,92 +222,166 @@ 9.190 change only the value of pitch. key-f/key-g moves it side to side 9.191 and changes yaw. key-v/key-b will spin the blue segment clockwise 9.192 and counterclockwise, and only affect roll." 9.193 - [] 9.194 - (let [hand (box 0.2 1 0.2 :position (Vector3f. 0 0 0) 9.195 - :mass 0 :color ColorRGBA/Green :name "hand") 9.196 - finger (box 0.2 1 0.2 :position (Vector3f. 0 2.4 0) 9.197 - :mass 1 :color ColorRGBA/Red :name "finger") 9.198 - joint-node (box 0.1 0.05 0.05 :color ColorRGBA/Yellow 9.199 - :position (Vector3f. 0 1.2 0) 9.200 - :rotation (doto (Quaternion.) 9.201 - (.fromAngleAxis 9.202 - (/ Math/PI 2) 9.203 - (Vector3f. 0 0 1))) 9.204 - :physical? false) 9.205 - joint (join-at-point hand finger (Vector3f. 0 1.2 0 )) 9.206 - creature (nodify [hand finger joint-node]) 9.207 - finger-control (.getControl finger RigidBodyControl) 9.208 - hand-control (.getControl hand RigidBodyControl)] 9.209 - 9.210 + ([] (test-proprioception false)) 9.211 + ([record?] 9.212 + (let [hand (box 0.2 1 0.2 :position (Vector3f. 0 0 0) 9.213 + :mass 0 :color ColorRGBA/Gray :name "hand") 9.214 + finger (box 0.2 1 0.2 :position (Vector3f. 0 2.4 0) 9.215 + :mass 1 9.216 + :color 9.217 + (ColorRGBA. (/ 184 255) (/ 127 255) (/ 201 255) 1) 9.218 + :name "finger") 9.219 + joint-node (box 0.1 0.05 0.05 :color ColorRGBA/Yellow 9.220 + :position (Vector3f. 0 1.2 0) 9.221 + :rotation (doto (Quaternion.) 9.222 + (.fromAngleAxis 9.223 + (/ Math/PI 2) 9.224 + (Vector3f. 0 0 1))) 9.225 + :physical? false) 9.226 + creature (nodify [hand finger joint-node]) 9.227 + finger-control (.getControl finger RigidBodyControl) 9.228 + hand-control (.getControl hand RigidBodyControl) 9.229 + joint (joint-dispatch {:type :point} hand-control finger-control 9.230 + (Vector3f. 0 1.2 0) 9.231 + (Vector3f. 0 -1.2 0) nil) 9.232 9.233 - (let 9.234 - ;; ******************************************* 9.235 - 9.236 - [floor (box 10 10 10 :position (Vector3f. 0 -15 0) 9.237 - :mass 0 :color ColorRGBA/Gray) 9.238 - 9.239 - root (nodify [creature floor]) 9.240 - prop (joint-proprioception creature joint-node) 9.241 - prop-view (proprioception-debug-window) 9.242 - 9.243 - controls 9.244 - (merge standard-debug-controls 9.245 - {"key-o" 9.246 - (fn [_ _] (.setEnabled finger-control true)) 9.247 - "key-p" 9.248 - (fn [_ _] (.setEnabled finger-control false)) 9.249 - "key-k" 9.250 - (fn [_ _] (.setEnabled hand-control true)) 9.251 - "key-l" 9.252 - (fn [_ _] (.setEnabled hand-control false)) 9.253 - "key-i" 9.254 - (fn [world _] (set-gravity world (Vector3f. 0 0 0))) 9.255 - "key-period" 9.256 - (fn [world _] 9.257 - (.setEnabled finger-control false) 9.258 - (.setEnabled hand-control false) 9.259 - (.rotate creature (doto (Quaternion.) 9.260 - (.fromAngleAxis 9.261 - (float (/ Math/PI 15)) 9.262 - (Vector3f. 0 0 -1)))) 9.263 - 9.264 - (.setEnabled finger-control true) 9.265 - (.setEnabled hand-control true) 9.266 - (set-gravity world (Vector3f. 0 0 0)) 9.267 - ) 9.268 - 9.269 - 9.270 - } 9.271 - ) 9.272 + root (nodify [creature]) 9.273 + prop (proprioception-kernel creature joint-node) 9.274 + prop-view (view-proprioception)] 9.275 + (.setCollisionGroup 9.276 + (.getControl hand RigidBodyControl) 9.277 + PhysicsCollisionObject/COLLISION_GROUP_NONE) 9.278 + (apply 9.279 + world 9.280 + (with-movement 9.281 + finger 9.282 + ["key-r" "key-t" "key-f" "key-g" "key-v" "key-b"] 9.283 + [1 1 10 10 10 10] 9.284 + [root 9.285 + standard-debug-controls 9.286 + (fn [world] 9.287 + (if record? 9.288 + (Capture/captureVideo 9.289 + world 9.290 + (File. "/home/r/proj/cortex/render/proprio/main-view"))) 9.291 + (.setTimer world (com.aurellem.capture.RatchetTimer. 60)) 9.292 + (set-gravity world (Vector3f. 0 0 0)) 9.293 + (enable-debug world) 9.294 + (light-up-everything world)) 9.295 + (fn [_ _] 9.296 + (prop-view 9.297 + (list (prop)) 9.298 + (if record? 9.299 + (File. "/home/r/proj/cortex/render/proprio/proprio"))))]))))) 9.300 +#+end_src 9.301 9.302 - ] 9.303 - (comment 9.304 - (.setCollisionGroup 9.305 - (.getControl hand RigidBodyControl) 9.306 - PhysicsCollisionObject/COLLISION_GROUP_NONE) 9.307 - ) 9.308 - (apply 9.309 - world 9.310 - (with-movement 9.311 - hand 9.312 - ["key-y" "key-u" "key-h" "key-j" "key-n" "key-m"] 9.313 - [10 10 10 10 1 1] 9.314 - (with-movement 9.315 - finger 9.316 - ["key-r" "key-t" "key-f" "key-g" "key-v" "key-b"] 9.317 - [1 1 10 10 10 10] 9.318 - [root 9.319 - controls 9.320 - (fn [world] 9.321 - (.setTimer world (com.aurellem.capture.RatchetTimer. 60)) 9.322 - (set-gravity world (Vector3f. 0 0 0)) 9.323 - (light-up-everything world)) 9.324 - (fn [_ _] (prop-view (list (prop))))])))))) 9.325 +#+results: test-proprioception 9.326 +: #'cortex.test.proprioception/test-proprioception 9.327 9.328 +* Video of Proprioception 9.329 + 9.330 +#+begin_html 9.331 +<div class="figure"> 9.332 +<center> 9.333 +<video controls="controls" width="550"> 9.334 + <source src="../video/test-proprioception.ogg" type="video/ogg" 9.335 + preload="none" poster="../images/aurellem-1280x480.png" /> 9.336 +</video> 9.337 +</center> 9.338 +<p>Proprioception in a simple creature. The proprioceptive readout is 9.339 + in the upper left corner of the screen.</p> 9.340 +</div> 9.341 +#+end_html 9.342 + 9.343 +** Generating the Proprioception Video 9.344 +#+name: magick6 9.345 +#+begin_src clojure 9.346 +(ns cortex.video.magick6 9.347 + (:import java.io.File) 9.348 + (:use clojure.contrib.shell-out)) 9.349 + 9.350 +(defn images [path] 9.351 + (sort (rest (file-seq (File. path))))) 9.352 + 9.353 +(def base "/home/r/proj/cortex/render/proprio/") 9.354 + 9.355 +(defn pics [file] 9.356 + (images (str base file))) 9.357 + 9.358 +(defn combine-images [] 9.359 + (let [main-view (pics "main-view") 9.360 + proprioception (pics "proprio/0") 9.361 + targets (map 9.362 + #(File. (str base "out/" (format "%07d.png" %))) 9.363 + (range 0 (count main-view)))] 9.364 + (dorun 9.365 + (pmap 9.366 + (comp 9.367 + (fn [[ main-view proprioception target]] 9.368 + (println target) 9.369 + (sh "convert" 9.370 + main-view 9.371 + proprioception "-geometry" "+20+20" "-composite" 9.372 + target)) 9.373 + (fn [& args] (map #(.getCanonicalPath %) args))) 9.374 + main-view proprioception targets)))) 9.375 #+end_src 9.376 9.377 +#+begin_src sh :results silent 9.378 +cd ~/proj/cortex/render/proprio 9.379 +ffmpeg -r 60 -i out/%07d.png -b:v 9000k -c:v libtheora \ 9.380 + test-proprioception.ogg 9.381 +#+end_src 9.382 + 9.383 +* Headers 9.384 +#+name: proprioception-header 9.385 +#+begin_src clojure 9.386 +(ns cortex.proprioception 9.387 + "Simulate the sense of proprioception (ability to detect the 9.388 + relative positions of body parts with repsect to other body parts) 9.389 + in jMonkeyEngine3. Reads specially prepared blender files to 9.390 + automatically generate proprioceptive senses." 9.391 + (:use (cortex world util sense body)) 9.392 + (:use clojure.contrib.def) 9.393 + (:import com.jme3.scene.Node) 9.394 + (:import java.awt.image.BufferedImage) 9.395 + (:import (com.jme3.math Vector3f Quaternion))) 9.396 +#+end_src 9.397 + 9.398 +#+name: test-proprioception-header 9.399 +#+begin_src clojure 9.400 +(ns cortex.test.proprioception 9.401 +(:import (com.aurellem.capture Capture RatchetTimer)) 9.402 +(:use (cortex util world proprioception body)) 9.403 +(:import java.io.File)) 9.404 +(cortex.import/mega-import-jme3) 9.405 +#+end_src 9.406 + 9.407 +* Source Listing 9.408 + - [[../src/cortex/proprioception.clj][cortex.proprioception]] 9.409 + - [[../src/cortex/test/touch.clj][cortex.test.proprioception]] 9.410 + - [[../src/cortex/video/magick6.clj][cortex.video.magick6]] 9.411 +#+html: <ul> <li> <a href="../org/proprioception.org">This org file</a> </li> </ul> 9.412 + - [[http://hg.bortreb.com ][source-repository]] 9.413 + 9.414 +* Next 9.415 + 9.416 +Next time, I'll give the Worm the power to [[./movement.org][move on it's own]]. 9.417 + 9.418 9.419 * COMMENT generate source 9.420 #+begin_src clojure :tangle ../src/cortex/proprioception.clj 9.421 +<<proprioception-header>> 9.422 +<<helpers>> 9.423 <<proprioception>> 9.424 +<<visualize>> 9.425 #+end_src 9.426 + 9.427 +#+begin_src clojure :tangle ../src/cortex/test/proprioception.clj 9.428 +<<test-proprioception-header>> 9.429 +<<test-proprioception>> 9.430 +#+end_src 9.431 + 9.432 +#+begin_src clojure :tangle ../src/cortex/video/magick6.clj 9.433 +<<magick6>> 9.434 +#+end_src
10.1 --- a/org/touch.org Mon Feb 13 21:53:28 2012 -0600 10.2 +++ b/org/touch.org Mon Feb 13 21:53:54 2012 -0600 10.3 @@ -122,7 +122,7 @@ 10.4 #+end_src 10.5 10.6 It is convienent to treat a =Triangle= as a vector of vectors, and a 10.7 -=Vector2f= and =Vector3f= as vectors of floats. (->vector3f) and 10.8 +=Vector2f= or =Vector3f= as vectors of floats. (->vector3f) and 10.9 (->triangle) undo the operations of =(vector3f-seq)= and 10.10 =(triangle-seq)=. If these classes implemented =Iterable= then =(seq)= 10.11 would work on them automitacally. 10.12 @@ -497,6 +497,7 @@ 10.13 I'll use a new creature --- a simple cube which has touch sensors 10.14 evenly distributed along each of its sides. 10.15 10.16 +#+name: test-touch-0 10.17 #+begin_src clojure 10.18 (in-ns 'cortex.test.touch) 10.19 10.20 @@ -504,10 +505,7 @@ 10.21 (load-blender-model "Models/test-touch/touch-cube.blend")) 10.22 #+end_src 10.23 10.24 -#+begin_html 10.25 -<br> 10.26 -#+end_html 10.27 - 10.28 +** The Touch Cube 10.29 #+begin_html 10.30 <div class="figure"> 10.31 <center> 10.32 @@ -527,12 +525,10 @@ 10.33 #+caption: The distribution of feelers along the touch-cube. The colors of the faces are irrelevant; only the white pixels specify feelers. 10.34 [[../images/touch-profile.png]] 10.35 10.36 +#+name: test-touch-1 10.37 #+begin_src clojure 10.38 (in-ns 'cortex.test.touch) 10.39 10.40 -(import com.aurellem.capture.Capture) 10.41 -(import java.io.File) 10.42 - 10.43 (defn test-basic-touch 10.44 ([] (test-basic-touch false)) 10.45 ([record?] 10.46 @@ -576,6 +572,7 @@ 10.47 #+end_html 10.48 10.49 ** Generating the Basic Touch Video 10.50 +#+name: magick4 10.51 #+begin_src clojure 10.52 (ns cortex.video.magick4 10.53 (:import java.io.File) 10.54 @@ -615,31 +612,88 @@ 10.55 background main-view touch targets)))) 10.56 #+end_src 10.57 10.58 +#+begin_src sh :results silent 10.59 +cd /home/r/proj/cortex/render/touch-cube/ 10.60 +ffmpeg -r 60 -i out/%07d.png -b:v 9000k -c:v libtheora basic-touch.ogg 10.61 +#+end_src 10.62 10.63 * Adding Touch to the Worm 10.64 10.65 -#+name: test-touch 10.66 +#+name: test-touch-2 10.67 #+begin_src clojure 10.68 -(ns cortex.test.touch 10.69 - (:use (cortex world util sense body touch)) 10.70 - (:use cortex.test.body)) 10.71 +(in-ns 'cortex.test.touch) 10.72 10.73 -(cortex.import/mega-import-jme3) 10.74 +(defn test-touch 10.75 + ([] (test-touch false)) 10.76 + ([record?] 10.77 + (let [the-worm (doto (worm) (body!)) 10.78 + touch (touch! the-worm) 10.79 + touch-display (view-touch)] 10.80 + (world 10.81 + (nodify [the-worm (floor)]) 10.82 + standard-debug-controls 10.83 + 10.84 + (fn [world] 10.85 + (if record? 10.86 + (Capture/captureVideo 10.87 + world 10.88 + (File. "/home/r/proj/cortex/render/worm-touch/main-view/"))) 10.89 + (speed-up world) 10.90 + (light-up-everything world)) 10.91 10.92 -(defn test-touch [] 10.93 - (let [the-worm (doto (worm) (body!)) 10.94 - touch (touch! the-worm) 10.95 - touch-display (view-touch)] 10.96 - (world (nodify [the-worm (floor)]) 10.97 - standard-debug-controls 10.98 - 10.99 - (fn [world] 10.100 - (speed-up world) 10.101 - (light-up-everything world)) 10.102 + (fn [world tpf] 10.103 + (touch-display 10.104 + (map #(% (.getRootNode world)) touch) 10.105 + (if record? 10.106 + (File. "/home/r/proj/cortex/render/worm-touch/touch/")))))))) 10.107 +#+end_src 10.108 10.109 - (fn [world tpf] 10.110 - (touch-display 10.111 - (map #(% (.getRootNode world)) touch)))))) 10.112 +** Worm Touch Demonstration 10.113 +#+begin_html 10.114 +<div class="figure"> 10.115 +<center> 10.116 +<video controls="controls" width="550"> 10.117 + <source src="../video/worm-touch.ogg" type="video/ogg" 10.118 + preload="none" poster="../images/aurellem-1280x480.png" /> 10.119 +</video> 10.120 +</center> 10.121 +<p>The worm responds to touch.</p> 10.122 +</div> 10.123 +#+end_html 10.124 + 10.125 + 10.126 +** Generating the Worm Touch Video 10.127 +#+name: magick5 10.128 +#+begin_src clojure 10.129 +(ns cortex.video.magick5 10.130 + (:import java.io.File) 10.131 + (:use clojure.contrib.shell-out)) 10.132 + 10.133 +(defn images [path] 10.134 + (sort (rest (file-seq (File. path))))) 10.135 + 10.136 +(def base "/home/r/proj/cortex/render/worm-touch/") 10.137 + 10.138 +(defn pics [file] 10.139 + (images (str base file))) 10.140 + 10.141 +(defn combine-images [] 10.142 + (let [main-view (pics "main-view") 10.143 + touch (pics "touch/0") 10.144 + targets (map 10.145 + #(File. (str base "out/" (format "%07d.png" %))) 10.146 + (range 0 (count main-view)))] 10.147 + (dorun 10.148 + (pmap 10.149 + (comp 10.150 + (fn [[ main-view touch target]] 10.151 + (println target) 10.152 + (sh "convert" 10.153 + main-view 10.154 + touch "-geometry" "+0+0" "-composite" 10.155 + target)) 10.156 + (fn [& args] (map #(.getCanonicalPath %) args))) 10.157 + main-view touch targets)))) 10.158 #+end_src 10.159 10.160 * Headers 10.161 @@ -661,9 +715,44 @@ 10.162 (:import (com.jme3.math Triangle Vector3f Vector2f Ray Matrix4f))) 10.163 #+end_src 10.164 10.165 +#+name: test-touch-header 10.166 +#+begin_src clojure 10.167 +(ns cortex.test.touch 10.168 + (:use (cortex world util sense body touch)) 10.169 + (:use cortex.test.body) 10.170 + (:import com.aurellem.capture.Capture) 10.171 + (:import java.io.File) 10.172 + (:import (com.jme3.math Vector3f ColorRGBA))) 10.173 +#+end_src 10.174 + 10.175 * Source Listing 10.176 + - [[../src/cortex/touch.clj][cortex.touch]] 10.177 + - [[../src/cortex/test/touch.clj][cortex.test.touch]] 10.178 + - [[../src/cortex/video/magick4.clj][cortex.video.magick4]] 10.179 + - [[../src/cortex/video/magick5.clj][cortex.video.magick5]] 10.180 +#+html: <ul> <li> <a href="../org/touch.org">This org file</a> </li> </ul> 10.181 + - [[http://hg.bortreb.com ][source-repository]] 10.182 + 10.183 * Next 10.184 +So far I've implemented simulated Vision, Hearing, and Touch, the most 10.185 +obvious and promiment senses that humans have. Smell and Taste shall 10.186 +remain unimplemented for now. This accounts for the "five senses" that 10.187 +feature so prominently in our lives. But humans have far more than the 10.188 +five main senses. There are internal chemical senses, pain (which is 10.189 +*not* the same as touch), heat sensitivity, and our sense of balance, 10.190 +among others. One extra sense is so important that I must implement it 10.191 +to have a hope of making creatures that can gracefully control their 10.192 +own bodies. It is Proprioception, which is the sense of the location 10.193 +of each body part in relation to the other body parts. 10.194 10.195 +Close your eyes, and touch your nose with your right index finger. How 10.196 +did you do it? You could not see your hand, and neither your hand nor 10.197 +your nose could use the sense of touch to guide the path of your hand. 10.198 +There are no sound cues, and Taste and Smell certainly don't provide 10.199 +any help. You know where your hand is without your other senses 10.200 +because of Proprioception. 10.201 + 10.202 +Onward to [[./proprioception.org][proprioception]]! 10.203 10.204 * COMMENT Code Generation 10.205 #+begin_src clojure :tangle ../src/cortex/touch.clj 10.206 @@ -678,11 +767,22 @@ 10.207 <<visualization>> 10.208 #+end_src 10.209 10.210 +#+begin_src clojure :tangle ../src/cortex/test/touch.clj 10.211 +<<test-touch-header>> 10.212 +<<test-touch-0>> 10.213 +<<test-touch-1>> 10.214 +<<test-touch-2>> 10.215 +#+end_src 10.216 10.217 -#+begin_src clojure :tangle ../src/cortex/test/touch.clj 10.218 -<<test-touch>> 10.219 +#+begin_src clojure :tangle ../src/cortex/video/magick4.clj 10.220 +<<magick4>> 10.221 #+end_src 10.222 10.223 +#+begin_src clojure :tangle ../src/cortex/video/magick5.clj 10.224 +<<magick5>> 10.225 +#+end_src 10.226 + 10.227 + 10.228 10.229 10.230