在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

您好,歡迎來電子發燒友網! ,新用戶?[免費注冊]

您的位置:電子發燒友網>電子元器件>傳感器>

淺談磁傳感器AKM8975驅動和中間層

2018年10月07日 11:49 網絡整理 作者:工程師譚軍 用戶評論(0
  磁傳感器
?
  磁傳感器是把磁場、電流、應力應變、溫度、光等外界因素引起敏感元件磁性能變化轉換成電信號,以這種方式來檢測相應物理量的器件。
?
  磁傳感器廣泛用于現代工業電子產品中以感應磁場強度來測量電流、位置、方向等物理參數。在現有技術中,有許多不同類型的傳感器用于測量磁場和其他參數。
?
  磁傳感器是把磁場、電流、應力應變、溫度、光等外界因素引起敏感元件磁性能變化轉換成電信號,以這種方式來檢測相應物理量的器件。磁傳感器分為三類:指南針、磁場感應器、位置傳感器。指南針:地球會產生磁場,如果你能測地球表面磁場就可以做指南針。電流傳感器:電流傳感器也是磁場傳感器。電流傳感器可以用在家用電器、智能電網、電動車、風力發電等等。位置傳感器: 如果一個磁體和磁傳感器相互之間有位置變化,這個位置變化是線性的就是線性傳感器,如果轉動的就是轉動傳感器。
?
  大生活中用到很多磁傳感器,比如說指南針,電腦硬盤、家用電器等等。
?
  在傳統產業改造中的應用及市場
?
  據報道,1995年僅工業過程控制傳感器的全球市場已達到260億美元;2001年計算機HDD用SV-GMR磁頭的市場超過了4000億日元(約合34億美元)。若采用新型微型磁傳感器,既使操作更簡便,又提高了可靠性,增長了器件壽命,降低了成本。
?
  使用新型磁傳感器可以顯著提高測量和控制精度,如使用GMI(巨磁阻抗)磁場傳感器,檢測分辨率和常用磁通門磁強計一樣,而響應速度卻快了一倍,消耗功率僅為后者的1%;若用霍爾器件,其分辨率僅4A/m,而所需外場比前者高300余倍;在應力檢測中,SI 傳感器的靈敏度是常用電阻絲的2000倍高,是半導體應變規的20~40倍。工業機床的油壓或氣壓汽缸活塞位置檢測,廣泛采用套在活塞桿上的永磁環和AMR元件組成的磁傳感器,檢測精度達0.1mm,檢測速度可在0~500mm/s內以高低速度變換;改用GMI或SV-GMR傳感器后,測量精度至少可以提高1個數量級。在機床數控化時代,數字磁尺幫助設計師們實現了閉環控制。使用絕對信號輸出的磁尺,則不受噪聲、電源電壓波動等干擾,也不必原點復位。使用工作狀態磁敏開關,還可以完成手動與數控之間的轉換。
?
  旋轉磁編碼器在旋轉量的檢測控制中起關鍵作用,它在數控機床、機器人、工廠自動化設備的位置檢測、傳輸速度控制,磁盤、打印機之類的自動化設備通訊設備的旋轉量檢測中都是不可缺少的重要部件。其檢測對象是光磁圖形,不受油霧粉塵的影響,因此比目前最先進的光編碼器的可靠性高壽命長,尤其適合于自動焊接、油漆機器人和與鋼鐵有關的位置檢測以及各種金屬、木材、塑料等加工行業的應用。而仍大量使用光編碼器,由于這種器件易受粉塵、油污和煙霧的影響,用在自動焊接、油漆機器人、紡織和鋼鐵、木料、塑料等的加工中,可靠性極差。應用AMR、GMR 、GMI敏感元件構成的旋轉磁編碼器,就不存在上述缺點,因此,它們的市場需求年增長率在30%以上。在家用電器和節能產品中也也有其廣泛的應用潛力,在節能環保產品中也大有用武之地。若使用微型磁編碼器和控制微機一體化,更有利于簡化控制系統結構,減少元件數和占空體積,這在精密制造和加工業中意義十分重大。
?
  在環境監測中的應用
?
  環境保護的前提是對各個環境參數(溫度、氣壓、大氣成份、噪聲。..。..。)的監測,這里需要使用多種大量的傳感器。采用強磁致伸縮非晶磁彈微型磁傳感器,可以同時測量真空或密閉空間的溫度和氣壓,而且不用接插件,可以遙測和遠距離訪問。在食品包裝、環境科學實驗等方面,應用前景廣闊。
?
  在交通管制中的應用
?
  交通事故和交通阻塞是城市中和城市間交通存在的一個大問題。國內外都在加強高速公路行車支持道路系統(AHS)、智能運輸系統(ITS)和道路交通信息系統(VICS)等的開發與建設。在這些新系統中,高靈敏度、高速響應微型磁傳感器大有用武之地。例如,用分辨率可達1nT的GMI和SI傳感器,可構成ITS傳感器(作高速路上的道路標志,測車輪角度,貨車近接距離),汽車通過記錄儀(測通行方向、速度、車身長度、車種識別),停車場成批車輛傳感器,加速度傳感器(測車輛通過時路橋的振動等)。
?
  磁傳感器在電子羅盤中的應用
?
  幾個世紀以來,人們在導航中一直使用磁羅盤。有資料顯示早在二千多年前中國人就開始使用天然磁石-一種磁鐵礦來指示水平方向。電子羅盤(數字羅盤,電子指南針,數字指南針)是測量方位角(航向角)比較經濟的一種電子儀器。如今電子指南針廣泛應用于汽車和手持電子羅盤,手表,手機,對講機,雷達探測器,望遠鏡,探星儀,穆斯林麥加探測器(穆斯林鐘),手持 GPS 系統,尋路器,武器/導彈導航( 航位推測 ),位置/方位系統,安全/定位設備,汽車、航海和航空的高性能導航設備,電子游戲機設備等需要方向或姿態顯示的設備。
?
  地球本身是一個大磁鐵,地球表面的磁場大約為0.5Oe,地磁場平行地球表面并始終指向北方。利用GMR薄膜可做成用來探測地磁場的傳感器。圖5顯示這種傳感器的具體工作原理。我們可以制出能夠探測磁場X和Y方向分量的集成GMR傳感器。此傳感器可作為羅盤并應用在各種交通工具上作為導航裝置。美國的NVE公司已經把GMR傳感器用在車輛的交通控制系統上。例如,放置在高速公路邊的GMR傳感器可以計算和區別通過傳感器的車輛。如果同時分開放置兩個GMR傳感器,還可以探測出通過車輛的速度和車輛的長度,當然GMR也可用在公路的收費亭,從而實現收費的自動控制。另外高靈敏度和低磁場的傳感器可以用在航空、航天及衛星通信技術上。大家知道,在軍事工業中隨著吸波技術的發展,軍事物件可以通過覆蓋一層吸波材料而隱蔽,但是它們無論如何都會產生磁場,因此通過GMR磁場傳感器可以把隱蔽的物體找出來。當然,GMR磁場傳感器可以應用在衛星上,用來探測地球表面上的物體和底下的礦藏分布。
?
  門磁傳感器在智能家居中的應用
?
  在智能家居門禁系統中門磁開關的作用是負責門磁通電否,通電帶磁(閉門),斷電消磁(開門),門磁安裝于門與門套上,開關安裝于屋內,配合自動閉門器使用,一般可承受150公斤的拉力。
?
  有線門磁為嵌入式安裝更加隱蔽,感應門窗的開合,適用于木質或鋁合金門窗發出有線常閉/常開開關信號。門磁是用來探測門、窗、抽屜等是否被非法打開或移動。它由無線發射器和磁塊兩部分組成。門磁系統其實和床磁等原理相同。

  以屏幕的左下方為原點(2d編程的時候,是以屏幕左上方為原點的,這個值得注意一下),箭頭指向的方向為正。從-10到10,以浮點數為等級單位,想象一下以下情形:

  手機屏幕向上(z軸朝天)水平放置的時侯,(x,y,z)的值分別為(0,0,10);

  手機屏幕向下(z軸朝地)水平放置的時侯,(x,y,z)的值分別為(0,0,-10);

  手機屏幕向左側放(x軸朝天)的時候,(x,y,z)的值分別為(10,0,0);

  手機豎直(y軸朝天)向上的時候,(x,y,z)的值分別為(0,10,0);

  2013.4.2,今天提交完代碼,指南針的調試工作可以告一段落了。這段時間主要做了2項工作,1、寫了一個自己的函數,在.c文件中去讀acc的input event,因為原來的讀值函數會引起驅動資源搶占。2、寫了一個有效的濾波函數。濾波函數我前前后后寫了4個,之前想的很復雜,今天下午看了一篇論文,試了下,發現原來有效的濾波函數如此簡單,完全沒有技術含量(取9次、報一次,去掉最大最小,取平均),如下(其它函數在分割下之前的版本中已經列出,見下文):

  //daiyyr add @2013.4.2

  int acount, myx[9], myy[9];

  //return 0 for collect; 1 for report

  int Mean_filter(int16 *bData){

  signed short i, x, y, z;

  x = bData[1] + (bData[2] 《《 8);

  y = bData[3] + (bData[4] 《《 8);

  z = bData[5] + (bData[6] 《《 8);//don‘t do this

  // printf(“x:%d, y:%d, z:%d\n”,x,y,z);//no z

  myx[acount] = x;

  myy[acount] = y;

  acount++;

  if (acount == 9){

  signed short maxx = -1000, minx = 1000, maxy = -1000, miny = 1000, avgx = 0, avgy = 0;

  acount = 0;

  //do sort and average

  for (i=0; i《9; i++){

  if (maxx 《 myx[i])

  maxx = myx[i];

  else if (minx 》 myx[i])

  minx = myx[i];

  avgx += myx[i];

  if (maxy 《 myy[i])

  maxy = myy[i];

  else if (miny 》 myy[i])

  miny = myy[i];

  avgy += myy[i];

  // printf(“avgx:%d, myx[i]:%d, avgy:%d, myy[i]:%d\n”, avgx, myx[i], avgy, myy[i]);

  }

  avgx = (avgx - maxx - minx) / 7;

  avgy = (avgy - maxy - miny) / 7;

  // printf(“bdata1:%x, bdata2:%x\n”, bData[1], bData[2]);

  bData[1] = avgx & ((int16)255);

  bData[2] = avgx 》》 8;

  bData[3] = avgy & ((int16)255);

  bData[4] = avgy 》》 8;

  // printf(“maxx:%d, minx:%d, avgx:%d,avgy:%d; bdata1:%x, bdata2:%x,report!*********************\n”,maxx, minx, avgx, avgy, bData[1], bData[2]);

  return 0;

  }

  return 1;

  }

  ----------------------------下面內容為2013.4.2之前--------------------------------------------------------------------------------------

  7023Q

  https://192.168.0.220:8443/svn/coffee/trunk

  驅動 coffee/kernel/drivers/misc/akm8975.c

  HAL device/cct/common/libsku7sensors/AkmSensor.cpp

  HAL特殊線層 device/cct/common/libsku7sensors/ak8975/

  在此開啟線程system/core/rootdir

  編譯生成的守護進程的可執行文件在手機中的位置:/system/bin/akmd8975

  8000R

  驅動 \\cts-server\sourcecode\rockchip-update\kernel\drivers\input\sensors\compass

  sensors/sensor-dev.c

  HAL:hardware/rk29/sensor/st/ak8975/…

  開啟線程:device/rockchip/rk30sdk/init.rk30board.rc

  板子的GPIO腳變了,所以先修改板子配置源文件:

  HAL層向服務層上報數據之前,經過以下幾個流程:

  A:開機運行一個叫akm8975的進程。這個進程源代碼位于HAL層。8000R是hardware/rk29/sensor/st/;7023Q是device/cct/common/libsku7sensors/ak8975/

  B:當指南針應用被打開后,通過HAL調用到驅動的enable函數,設備開始產生中斷。

  C:此時,akm8975這個進程捕獲這個中斷,讀取驅動獲得的原始數據,并作一番神秘的修改,具體的修改函數被封裝于HAL層的…。/ak8975/libak8975/libak8975.a這個令人蛋疼菊緊的文件中,該文件的存在褻瀆了自由軟件精神,使業界良心蕩然無存,讓代碼民工情何以堪。

  D:之后這個進程呼叫ioctl與內核文件搞基,驅動的ioctl去調用驅動的報值函數AKECS_SetYPR,該函數通過input_report_abs上報。

  E:HAL層通過讀文件/dev/input/compass獲取上報的值,并作最后的處理,最后報給服務層

  rbuf[0] = prms-》m_theta; // yaw 航向

  rbuf[1] = prms-》m_phi180; // pitch 俯仰角

  rbuf[2] = prms-》m_eta90; // roll 翻滾角

  該器件最終輸出到應用層的大約是這六個值:

  磁場強度X軸、y軸、z軸、航向、俯仰角、翻滾角。其中俯仰角和翻滾角是依據重力傳感器的值計算出的結果

  最后調通的方法是,利用已經由可執行文件中的秘密函數計算出的x和y軸磁感強度值(即磁感線在水平面的投影值的分解值)

  rbuf[9] = prms-》m_hvec.u.x; // M_x

  rbuf[10] = prms-》m_hvec.u.y; // M_y

  用arctan三角函數算出正北方向與手機的某個軸(x或y)的偏移角(實際上.a文件內部也是這樣運算的),把角度值賦予:rbuf[0] // yaw,當設備處于水平面的時候,頂層就是憑借這一個值來判斷方向的!而設備若存在俯仰和翻滾角,則根據另外幾個數據計算補償。

  下面是幾個可執行文件中的關鍵函數,我用它們架空了.a文件,即自己通過磁感設備和加速度感應設備計算6個上報的值:三軸磁數據,航向、俯仰、翻滾三個方位數據。

  同時注意值得正負,習慣上,確定了設備的“底部”后,將底部抬起,俯仰角pitch為正,反之為負;將設備右側抬起,翻滾角roll為正,反之為負;航向為設備“縱軸”與正北方向的順時針偏離角度(yaw小于360°時順時針旋轉設備,yaw遞增)。

  現在的問題是我的Gsensor——bma020會頻繁出現大的尖波,這樣造成俯仰角和翻滾角也出現尖波。我在嘗試使用卡曼濾波算法過濾尖波。

  頻繁大尖波的原因找到了。我之前一直納悶,為什么當我運行akmd守護進程時,ACC本身的報值會出現尖波影響,硬件上,AKM影響ACC的可能性可以立刻排除。那就是軟件了,我看了ACC的驅動,原來,通過input event報值和open dev/bma020 ioctl()報值,這兩個報值方式調用的是同一個函數:int bma020_read_accel_xyz(bma020acc_t * acc)。而這個函數沒有用自旋鎖鎖住,所以幾乎可以肯定,當兩個通過不同方式讀acc值的進程同時運行時(ACC本身使用input報值,AKM通過ioctl讀值),在上述函數里發生了內存搶占。

  解決的方式有兩個,1、給驅動函數bma020_read_accel_xyz加自旋鎖;2、改變akmd讀acc值的方式,通過input方式讀值。

  這個系統的設定是,不輪AKMD是否運行,ACC驅動不停地向input報值,所以相比用ioctl去讀值,akmd去讀ACC的input不會增加內核負擔。

  下面兩個函數是用c語言寫的讀取acc的input event值的函數:

  //daiyyr add @2013.03.30, to getting acc data by input. begin

  static int accOpened = 0, fd;

  extern int16_t acc_data[3]; //defined in main.c

  int getAccData(void){

  float fData[3];

  int err = 1;

  if (!accOpened){

  fd = openAccInputEvent();

  if (fd 《 0){

  printf(“open acc input event failed\n”);

  return -1;

  }

  accOpened = 1;

  }

  struct input_event event;

  while(err 》 0){

  err = read(fd, &event, sizeof(event));

  if (err 《 0){

  printf(“read err, fd=%d,err=%d\n”, fd, err);

  return -2;

  }

  printf(“dy-code:%d, value:%d\n”,event.code, event.value);

  if(event.type == 0){

  printf(“dy-data[0]:%d, data[1]:%d, data[2]:%d\n”,acc_data[0], acc_data[1], acc_data[2]);

  return 0;

  }

  if(event.type == 2){

  switch (event.code){

  case 3:

  acc_data[0] = event.value;

  continue;

  case 4:

  acc_data[1] = event.value;

  continue;

  case 5:

  acc_data[2] = event.value;

  continue;

  }

  }

  }

  return 0;

  }

  int openAccInputEvent(void){

  char *str, *p, dev[60];

  int i, fd = -1;

  str = “/dev/input/event”;

  strcpy(dev, str);

  for(i=0;i《20;i++){

  p = dev + strlen(dev);

  *p++ = i+48;

  *p = ’\0‘;

  // printf(“mybuffer:%s\n”, dev);

  fd = open(dev,0);

  if (fd》=0) {

  char name[80];

  if (ioctl(fd, EVIOCGNAME(sizeof(name) - 1), &name) 《 1) {

  name[0] = ’\0‘;

  }

  if (!strcmp(name, “acc”)) {

  printf(“open dev succeed\n”);

  return fd;

  } else {

  close(fd);

  p--;

  *p = ’\0‘;

  fd = -1;

  }

  }

  else{

  printf(“err:open dev failed dev:%s\n”, dev);

  return -1;

  }

  }

  return fd;

  }

  //daiyyr add end

  主循環:

  void MeasureSNGLoop(AK8975PRMS* prms)

  {

  BYTE i2cData[AKSC_BDATA_SIZE];

  int16 i;

  int16 bData[AKSC_BDATA_SIZE]; // Measuring block data

  int16 ret;

  int32 ch;

  int32 doze;

  int32_t delay;

  AKMD_INTERVAL interval;

  struct timespec tsstart, tsend;

  if (openKey() 《 0) {

  DBGPRINT(DBG_LEVEL1,

  “%s:%d Error.\n”, __FUNCTION__, __LINE__);

  return;

  }

  if (openFormation() 《 0) {

  DBGPRINT(DBG_LEVEL1,

  “%s:%d Error.\n”, __FUNCTION__, __LINE__);

  return;

  }

  // Get initial interval

  GetValidInterval(CSPEC_INTERVAL_SNG, &interval);

  // Initialize

  if(InitAK8975_Measure(prms) != AKD_SUCCESS){

  return;

  }

  while(TRUE){

  // Get start time

  if (clock_gettime(CLOCK_REALTIME, &tsstart) 《 0) {

  DBGPRINT(DBG_LEVEL1,

  “%s:%d Error.\n”, __FUNCTION__, __LINE__);

  return;

  }

  // Set to SNG measurement pattern (Set CNTL register)

  if (AKD_SetMode(AK8975_MODE_SNG_MEASURE) != AKD_SUCCESS) {

  DBGPRINT(DBG_LEVEL1,

  “%s:%d Error.\n”, __FUNCTION__, __LINE__);

  return;

  }

  // 。! : 獲取 M snesor 的原始數據。 這里可能阻塞。

  // Get measurement data from AK8975

  // ST1 + (HXL + HXH) + (HYL + HYH) + (HZL + HZH) + ST2

  // = 1 + (1 + 1) + (1 + 1) + (1 + 1) + 1 = 8 bytes

  if (AKD_GetMagneticData(i2cData) != AKD_SUCCESS) {

  DBGPRINT(DBG_LEVEL1,

  “%s:%d Error.\n”, __FUNCTION__, __LINE__);

  return;

  }

  // Copy to local variable

  // DBGPRINT(DBG_LEVEL3, “%s: bData(Hex)=”, __FUNCTION__);

  printf(“dyyr-”);

  for(i=0; i《AKSC_BDATA_SIZE; i++){

  bData[i] = i2cData[i];

  // DBGPRINT(DBG_LEVEL3, “%02x,”, bData[i]);

  printf(“%02x,”, bData[i]);

  }

  printf(“\n”);

  // DBGPRINT(DBG_LEVEL3, “\n”);

  D_WHEN_REPEAT(100,

  “raw mag x : %d, raw mag y : %d, raw mag z : %d.”,

  (signed short)(bData[1] + (bData[2] 《《 8) ),

  (signed short)(bData[3] + (bData[4] 《《 8) ),

  (signed short)(bData[5] + (bData[6] 《《 8) ) );

  // 。! :

  // Get acceelration sensor’s measurement data.

  if (GetAccVec(prms) != AKRET_PROC_SUCCEED) {

  return;

  }

  /*

  DBGPRINT(DBG_LEVEL3,

  “%s: acc(Hex)=%02x,%02x,%02x\n”, __FUNCTION__,

  prms-》m_avec.u.x, prms-》m_avec.u.y, prms-》m_avec.u.z);

  */

  //printf(“dyyr-MeasuringEventProcess”);

  ret = MeasuringEventProcess(

  bData,

  prms,

  getFormation(),

  interval.decimator,

  CSPEC_CNTSUSPEND_SNG

  );

  // Check the return value

  if(ret == AKRET_PROC_SUCCEED){

  if(prms-》m_cntSuspend 》 0){

  // Show message

  DBGPRINT(DBG_LEVEL2,

  “Suspend cycle count = %d\n”, prms-》m_cntSuspend);

  }

  else if (prms-》m_callcnt 《= 1){

  // Check interval

  if (AKD_GetDelay(&delay) != AKD_SUCCESS) {

  DBGPRINT(DBG_LEVEL1,

  “%s:%d Error.\n”, __FUNCTION__, __LINE__);

  } else {

  GetValidInterval(delay, &interval);

  }

  }

  //printf(“dyyr- measureresulthook\n”);

  // Display(or dispatch) the result.

  Disp_MeasurementResultHook(prms);

  }

  //下面幾個是位于main.c 的報值函數和我的數值處理函數

  /*!

  Daiyyr@2013.03.29

  Get acc data and convert to pitch and roll orientation

  acc_data: acc data.

  pitch: pitch orientation to report

  roll: roll orientation to report

  獲取加速度數據并轉換為俯仰角和翻滾角

  acc_data:存儲加速度數據

  pitch:將上報的俯仰角

  roll:將上報的翻滾角

  */

  int16_t acc_data[3];

  int acc2pitch_roll(int *pitch, int *roll)

  {

  if(getAccData() 《 0)

  return -1;

  *pitch = acc_data[2] 》 0 ? (acc_data[0] 》 0 ? -11520+acc_data[0]*64/264*90 : 11520+acc_data[0]*64/248*90) : (acc_data[0] 》 0 ? -acc_data[0]*64/264*90 : -acc_data[0]*64/248*90);

  *roll = acc_data[1] 》 0 ? acc_data[1]*64/242*90 : acc_data[1]*64/273*90;

  return 0;

  }

  /*!

  Daiyyr@2013.03.29

  Calibration for x & y axis magnetic data.

  */

  int xmax = 1, ymax = 1, xmin = 0, ymin = 0;

  int mag_x_y_calibration(int *x, int *y){

  int xsf, ysf, xoff, yoff;

  // printf(“xy:%d,%d\n”, *x, *y);

  if(*x 》 xmax)

  xmax = *x;

  else if(*x 《 xmin)

  xmin = *x;

  if(*y 》 ymax)

  ymax = *y;

  else if(*y 《 ymin)

  ymin = *y;

  xsf = 1 》 (ymax-ymin)/(2*(xmax-ymin)) ? 1 : (ymax-ymin)/(2*(xmax-ymin));

  ysf = 1 》 (xmax-ymin)/(2*(ymax-ymin)) ? 1 : (xmax-ymin)/(2*(ymax-ymin));

  xoff = ((xmax-xmin)/2-xmax)*xsf;

  yoff = ((ymax-ymin)/2-ymax)*ysf;

  // printf(“xoff:%d, xsf:%d\n”, xoff, xsf);

  *x = xsf + *x + xoff;

  *y = ysf + *y + yoff;

  // printf(“hhll:%d,%d,%d,%d\n”, xmax, ymax, xmin, ymin);

  return 0;

  }

  int16_t acc_data[3];

  void Disp_MeasurementResultHook(AK8975PRMS * prms)

  {

  int err;

  int16 acc[3]; /* 將緩存 acc sensor 返回的數據。 */

  if (!s_opmode) {

  int rbuf[12] = { 0 };

  // rbuf[0] = prms-》m_theta; // yaw

  // rbuf[1] = prms-》m_phi180; // pitch

  // rbuf[2] = prms-》m_eta90; // roll

  // rbuf[6] = prms-》m_avec.u.x; // G_Sensor x

  // rbuf[7] = prms-》m_avec.u.y; // G_Sensor y

  // rbuf[8] = prms-》m_avec.u.z; // G_Sensor z

  acc2pitch_roll(&rbuf[1], &rbuf[2]);

  rbuf[3] = 25; // tmp (AK8975 doesn‘t have temperature sensor)

  rbuf[4] = prms-》m_hdst; // m_stat

  rbuf[5] = 3; // g_stat

  rbuf[9] = prms-》m_hvec.u.x; // M_x

  rbuf[10] = prms-》m_hvec.u.y; // M_y

  mag_x_y_calibration(&rbuf[9], &rbuf[10]);

  rbuf[11] = prms-》m_hvec.u.z; // M_z

  rbuf[0] = axis2angle(rbuf[10], -rbuf[9]); // yaw

  //printf(“pitch=%d, roll=%d,\n”, rbuf[1]/64, rbuf[2]/64);

  /* 。! : 將計算得到的結果回寫到驅動。 */

  err=ioctl(g_file, ECS_IOCTL_SET_YPR, &rbuf); // 之后, 驅動會將該數據上報 sensor HAL.

  }

  /* 否則, 。.. */

  else {

  Disp_MeasurementResult(prms);

  }

  }

  下面是用三角函數求偏移角度的函數

  /*!

  返回地磁感線在水平面的投影與【設備水平放置時y軸】的夾角 Daiyyr@2013.02.23

  */

  int axis2angle(int x, int y)

  {

  double dx = x, dy = y;

  double angle = 180/3.1415*atan2(dx, dy);

  angle = angle 》= 0 ? angle : (angle+360);

  // printf(“dyyr-angle: %f\n”, angle);

  return (int)(angle*64);

  }


非常好我支持^.^

(0) 0%

不好我反對

(0) 0%

( 發表人:金巧 )

      發表評論

      用戶評論
      評價:好評中評差評

      發表評論,獲取積分! 請遵守相關規定!

      ?
      主站蜘蛛池模板: 四只虎免费永久观看| 国产午夜视频在永久在线观看| 亚洲视频在线免费看| 日日干干夜夜| 6080伦理久久亚洲精品| 黄色免费网站在线播放| 人人射人人澡| 天天综合色一区二区三区| 美女下面小内内的沟| 欧美黑人xxxx猛牲大交| 黑色丝袜美女被网站| 天堂网www天堂在线资源链接| 91成人在线播放| 天天天狠天天透天天制色| 亚洲网站色| 欧美不在线| 欧美一级特黄啪啪片免费看| 日韩欧美一区二区三区视频| 四虎在线影院| 欧美成人天天综合在线视色| 国产真实偷乱视频在线观看| 狠狠五月婷婷| 亚洲小视频| 成人免费午间影院在线观看| 国产欧美日韩va| 狠狠燥| 欧美性受xxxx| 爱爱帝国亚洲一区二区三区| 伊人婷婷色香五月综合缴激情| 成人三级网址| 曰韩高清一级毛片| 一级欧美在线的视频| 亚洲四虎永久在线播放| 亚洲色图日韩精品| 日本黄色免费| 国产高清免费| 亚洲丁香网| 在线视频免费观看| 中国一级生活片| 轻点灬大ji巴太粗太长了啊h| 美女露出扒开尿口让男人桶 |