llc_c_ac.c 35 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439
  1. /*
  2. * llc_c_ac.c - actions performed during connection state transition.
  3. *
  4. * Description:
  5. * Functions in this module are implementation of connection component actions
  6. * Details of actions can be found in IEEE-802.2 standard document.
  7. * All functions have one connection and one event as input argument. All of
  8. * them return 0 On success and 1 otherwise.
  9. *
  10. * Copyright (c) 1997 by Procom Technology, Inc.
  11. * 2001-2003 by Arnaldo Carvalho de Melo <acme@conectiva.com.br>
  12. *
  13. * This program can be redistributed or modified under the terms of the
  14. * GNU General Public License as published by the Free Software Foundation.
  15. * This program is distributed without any warranty or implied warranty
  16. * of merchantability or fitness for a particular purpose.
  17. *
  18. * See the GNU General Public License for more details.
  19. */
  20. #include <linux/netdevice.h>
  21. #include <net/llc_conn.h>
  22. #include <net/llc_sap.h>
  23. #include <net/sock.h>
  24. #include <net/llc_c_ev.h>
  25. #include <net/llc_c_ac.h>
  26. #include <net/llc_c_st.h>
  27. #include <net/llc_pdu.h>
  28. #include <net/llc.h>
  29. #include "llc_output.h"
  30. static int llc_conn_ac_inc_vs_by_1(struct sock *sk, struct sk_buff *skb);
  31. static void llc_process_tmr_ev(struct sock *sk, struct sk_buff *skb);
  32. static int llc_conn_ac_data_confirm(struct sock *sk, struct sk_buff *ev);
  33. static int llc_conn_ac_inc_npta_value(struct sock *sk, struct sk_buff *skb);
  34. static int llc_conn_ac_send_rr_rsp_f_set_ackpf(struct sock *sk,
  35. struct sk_buff *skb);
  36. static int llc_conn_ac_set_p_flag_1(struct sock *sk, struct sk_buff *skb);
  37. #define INCORRECT 0
  38. int llc_conn_ac_clear_remote_busy(struct sock *sk, struct sk_buff *skb)
  39. {
  40. struct llc_sock *llc = llc_sk(sk);
  41. if (llc->remote_busy_flag) {
  42. u8 nr;
  43. struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
  44. llc->remote_busy_flag = 0;
  45. del_timer(&llc->busy_state_timer.timer);
  46. nr = LLC_I_GET_NR(pdu);
  47. llc_conn_resend_i_pdu_as_cmd(sk, nr, 0);
  48. }
  49. return 0;
  50. }
  51. int llc_conn_ac_conn_ind(struct sock *sk, struct sk_buff *skb)
  52. {
  53. struct llc_conn_state_ev *ev = llc_conn_ev(skb);
  54. ev->ind_prim = LLC_CONN_PRIM;
  55. return 0;
  56. }
  57. int llc_conn_ac_conn_confirm(struct sock *sk, struct sk_buff *skb)
  58. {
  59. struct llc_conn_state_ev *ev = llc_conn_ev(skb);
  60. ev->cfm_prim = LLC_CONN_PRIM;
  61. return 0;
  62. }
  63. static int llc_conn_ac_data_confirm(struct sock *sk, struct sk_buff *skb)
  64. {
  65. struct llc_conn_state_ev *ev = llc_conn_ev(skb);
  66. ev->cfm_prim = LLC_DATA_PRIM;
  67. return 0;
  68. }
  69. int llc_conn_ac_data_ind(struct sock *sk, struct sk_buff *skb)
  70. {
  71. llc_conn_rtn_pdu(sk, skb);
  72. return 0;
  73. }
  74. int llc_conn_ac_disc_ind(struct sock *sk, struct sk_buff *skb)
  75. {
  76. struct llc_conn_state_ev *ev = llc_conn_ev(skb);
  77. u8 reason = 0;
  78. int rc = 0;
  79. if (ev->type == LLC_CONN_EV_TYPE_PDU) {
  80. struct llc_pdu_un *pdu = llc_pdu_un_hdr(skb);
  81. if (LLC_PDU_IS_RSP(pdu) &&
  82. LLC_PDU_TYPE_IS_U(pdu) &&
  83. LLC_U_PDU_RSP(pdu) == LLC_2_PDU_RSP_DM)
  84. reason = LLC_DISC_REASON_RX_DM_RSP_PDU;
  85. else if (LLC_PDU_IS_CMD(pdu) &&
  86. LLC_PDU_TYPE_IS_U(pdu) &&
  87. LLC_U_PDU_CMD(pdu) == LLC_2_PDU_CMD_DISC)
  88. reason = LLC_DISC_REASON_RX_DISC_CMD_PDU;
  89. } else if (ev->type == LLC_CONN_EV_TYPE_ACK_TMR)
  90. reason = LLC_DISC_REASON_ACK_TMR_EXP;
  91. else
  92. rc = -EINVAL;
  93. if (!rc) {
  94. ev->reason = reason;
  95. ev->ind_prim = LLC_DISC_PRIM;
  96. }
  97. return rc;
  98. }
  99. int llc_conn_ac_disc_confirm(struct sock *sk, struct sk_buff *skb)
  100. {
  101. struct llc_conn_state_ev *ev = llc_conn_ev(skb);
  102. ev->reason = ev->status;
  103. ev->cfm_prim = LLC_DISC_PRIM;
  104. return 0;
  105. }
  106. int llc_conn_ac_rst_ind(struct sock *sk, struct sk_buff *skb)
  107. {
  108. u8 reason = 0;
  109. int rc = 1;
  110. struct llc_conn_state_ev *ev = llc_conn_ev(skb);
  111. struct llc_pdu_un *pdu = llc_pdu_un_hdr(skb);
  112. struct llc_sock *llc = llc_sk(sk);
  113. switch (ev->type) {
  114. case LLC_CONN_EV_TYPE_PDU:
  115. if (LLC_PDU_IS_RSP(pdu) &&
  116. LLC_PDU_TYPE_IS_U(pdu) &&
  117. LLC_U_PDU_RSP(pdu) == LLC_2_PDU_RSP_FRMR) {
  118. reason = LLC_RESET_REASON_LOCAL;
  119. rc = 0;
  120. } else if (LLC_PDU_IS_CMD(pdu) &&
  121. LLC_PDU_TYPE_IS_U(pdu) &&
  122. LLC_U_PDU_CMD(pdu) == LLC_2_PDU_CMD_SABME) {
  123. reason = LLC_RESET_REASON_REMOTE;
  124. rc = 0;
  125. }
  126. break;
  127. case LLC_CONN_EV_TYPE_ACK_TMR:
  128. case LLC_CONN_EV_TYPE_P_TMR:
  129. case LLC_CONN_EV_TYPE_REJ_TMR:
  130. case LLC_CONN_EV_TYPE_BUSY_TMR:
  131. if (llc->retry_count > llc->n2) {
  132. reason = LLC_RESET_REASON_LOCAL;
  133. rc = 0;
  134. }
  135. break;
  136. }
  137. if (!rc) {
  138. ev->reason = reason;
  139. ev->ind_prim = LLC_RESET_PRIM;
  140. }
  141. return rc;
  142. }
  143. int llc_conn_ac_rst_confirm(struct sock *sk, struct sk_buff *skb)
  144. {
  145. struct llc_conn_state_ev *ev = llc_conn_ev(skb);
  146. ev->reason = 0;
  147. ev->cfm_prim = LLC_RESET_PRIM;
  148. return 0;
  149. }
  150. int llc_conn_ac_clear_remote_busy_if_f_eq_1(struct sock *sk,
  151. struct sk_buff *skb)
  152. {
  153. struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
  154. if (LLC_PDU_IS_RSP(pdu) &&
  155. LLC_PDU_TYPE_IS_I(pdu) &&
  156. LLC_I_PF_IS_1(pdu) && llc_sk(sk)->ack_pf)
  157. llc_conn_ac_clear_remote_busy(sk, skb);
  158. return 0;
  159. }
  160. int llc_conn_ac_stop_rej_tmr_if_data_flag_eq_2(struct sock *sk,
  161. struct sk_buff *skb)
  162. {
  163. struct llc_sock *llc = llc_sk(sk);
  164. if (llc->data_flag == 2)
  165. del_timer(&llc->rej_sent_timer.timer);
  166. return 0;
  167. }
  168. int llc_conn_ac_send_disc_cmd_p_set_x(struct sock *sk, struct sk_buff *skb)
  169. {
  170. int rc = -ENOBUFS;
  171. struct llc_sock *llc = llc_sk(sk);
  172. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  173. if (nskb) {
  174. struct llc_sap *sap = llc->sap;
  175. llc_pdu_header_init(nskb, LLC_PDU_TYPE_U, sap->laddr.lsap,
  176. llc->daddr.lsap, LLC_PDU_CMD);
  177. llc_pdu_init_as_disc_cmd(nskb, 1);
  178. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  179. if (unlikely(rc))
  180. goto free;
  181. llc_conn_send_pdu(sk, nskb);
  182. llc_conn_ac_set_p_flag_1(sk, skb);
  183. }
  184. out:
  185. return rc;
  186. free:
  187. kfree_skb(nskb);
  188. goto out;
  189. }
  190. int llc_conn_ac_send_dm_rsp_f_set_p(struct sock *sk, struct sk_buff *skb)
  191. {
  192. int rc = -ENOBUFS;
  193. struct llc_sock *llc = llc_sk(sk);
  194. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  195. if (nskb) {
  196. struct llc_sap *sap = llc->sap;
  197. u8 f_bit;
  198. llc_pdu_decode_pf_bit(skb, &f_bit);
  199. llc_pdu_header_init(nskb, LLC_PDU_TYPE_U, sap->laddr.lsap,
  200. llc->daddr.lsap, LLC_PDU_RSP);
  201. llc_pdu_init_as_dm_rsp(nskb, f_bit);
  202. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  203. if (unlikely(rc))
  204. goto free;
  205. llc_conn_send_pdu(sk, nskb);
  206. }
  207. out:
  208. return rc;
  209. free:
  210. kfree_skb(nskb);
  211. goto out;
  212. }
  213. int llc_conn_ac_send_dm_rsp_f_set_1(struct sock *sk, struct sk_buff *skb)
  214. {
  215. int rc = -ENOBUFS;
  216. struct llc_sock *llc = llc_sk(sk);
  217. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  218. if (nskb) {
  219. struct llc_sap *sap = llc->sap;
  220. llc_pdu_header_init(nskb, LLC_PDU_TYPE_U, sap->laddr.lsap,
  221. llc->daddr.lsap, LLC_PDU_RSP);
  222. llc_pdu_init_as_dm_rsp(nskb, 1);
  223. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  224. if (unlikely(rc))
  225. goto free;
  226. llc_conn_send_pdu(sk, nskb);
  227. }
  228. out:
  229. return rc;
  230. free:
  231. kfree_skb(nskb);
  232. goto out;
  233. }
  234. int llc_conn_ac_send_frmr_rsp_f_set_x(struct sock *sk, struct sk_buff *skb)
  235. {
  236. u8 f_bit;
  237. int rc = -ENOBUFS;
  238. struct sk_buff *nskb;
  239. struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
  240. struct llc_sock *llc = llc_sk(sk);
  241. llc->rx_pdu_hdr = *((u32 *)pdu);
  242. if (LLC_PDU_IS_CMD(pdu))
  243. llc_pdu_decode_pf_bit(skb, &f_bit);
  244. else
  245. f_bit = 0;
  246. nskb = llc_alloc_frame(sk, llc->dev);
  247. if (nskb) {
  248. struct llc_sap *sap = llc->sap;
  249. llc_pdu_header_init(nskb, LLC_PDU_TYPE_U, sap->laddr.lsap,
  250. llc->daddr.lsap, LLC_PDU_RSP);
  251. llc_pdu_init_as_frmr_rsp(nskb, pdu, f_bit, llc->vS,
  252. llc->vR, INCORRECT);
  253. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  254. if (unlikely(rc))
  255. goto free;
  256. llc_conn_send_pdu(sk, nskb);
  257. }
  258. out:
  259. return rc;
  260. free:
  261. kfree_skb(nskb);
  262. goto out;
  263. }
  264. int llc_conn_ac_resend_frmr_rsp_f_set_0(struct sock *sk, struct sk_buff *skb)
  265. {
  266. int rc = -ENOBUFS;
  267. struct llc_sock *llc = llc_sk(sk);
  268. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  269. if (nskb) {
  270. struct llc_sap *sap = llc->sap;
  271. struct llc_pdu_sn *pdu = (struct llc_pdu_sn *)&llc->rx_pdu_hdr;
  272. llc_pdu_header_init(nskb, LLC_PDU_TYPE_U, sap->laddr.lsap,
  273. llc->daddr.lsap, LLC_PDU_RSP);
  274. llc_pdu_init_as_frmr_rsp(nskb, pdu, 0, llc->vS,
  275. llc->vR, INCORRECT);
  276. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  277. if (unlikely(rc))
  278. goto free;
  279. llc_conn_send_pdu(sk, nskb);
  280. }
  281. out:
  282. return rc;
  283. free:
  284. kfree_skb(nskb);
  285. goto out;
  286. }
  287. int llc_conn_ac_resend_frmr_rsp_f_set_p(struct sock *sk, struct sk_buff *skb)
  288. {
  289. u8 f_bit;
  290. int rc = -ENOBUFS;
  291. struct sk_buff *nskb;
  292. struct llc_sock *llc = llc_sk(sk);
  293. llc_pdu_decode_pf_bit(skb, &f_bit);
  294. nskb = llc_alloc_frame(sk, llc->dev);
  295. if (nskb) {
  296. struct llc_sap *sap = llc->sap;
  297. struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
  298. llc_pdu_header_init(nskb, LLC_PDU_TYPE_U, sap->laddr.lsap,
  299. llc->daddr.lsap, LLC_PDU_RSP);
  300. llc_pdu_init_as_frmr_rsp(nskb, pdu, f_bit, llc->vS,
  301. llc->vR, INCORRECT);
  302. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  303. if (unlikely(rc))
  304. goto free;
  305. llc_conn_send_pdu(sk, nskb);
  306. }
  307. out:
  308. return rc;
  309. free:
  310. kfree_skb(nskb);
  311. goto out;
  312. }
  313. int llc_conn_ac_send_i_cmd_p_set_1(struct sock *sk, struct sk_buff *skb)
  314. {
  315. int rc;
  316. struct llc_sock *llc = llc_sk(sk);
  317. struct llc_sap *sap = llc->sap;
  318. llc_pdu_header_init(skb, LLC_PDU_TYPE_I, sap->laddr.lsap,
  319. llc->daddr.lsap, LLC_PDU_CMD);
  320. llc_pdu_init_as_i_cmd(skb, 1, llc->vS, llc->vR);
  321. rc = llc_mac_hdr_init(skb, llc->dev->dev_addr, llc->daddr.mac);
  322. if (likely(!rc)) {
  323. llc_conn_send_pdu(sk, skb);
  324. llc_conn_ac_inc_vs_by_1(sk, skb);
  325. }
  326. return rc;
  327. }
  328. static int llc_conn_ac_send_i_cmd_p_set_0(struct sock *sk, struct sk_buff *skb)
  329. {
  330. int rc;
  331. struct llc_sock *llc = llc_sk(sk);
  332. struct llc_sap *sap = llc->sap;
  333. llc_pdu_header_init(skb, LLC_PDU_TYPE_I, sap->laddr.lsap,
  334. llc->daddr.lsap, LLC_PDU_CMD);
  335. llc_pdu_init_as_i_cmd(skb, 0, llc->vS, llc->vR);
  336. rc = llc_mac_hdr_init(skb, llc->dev->dev_addr, llc->daddr.mac);
  337. if (likely(!rc)) {
  338. llc_conn_send_pdu(sk, skb);
  339. llc_conn_ac_inc_vs_by_1(sk, skb);
  340. }
  341. return rc;
  342. }
  343. int llc_conn_ac_send_i_xxx_x_set_0(struct sock *sk, struct sk_buff *skb)
  344. {
  345. int rc;
  346. struct llc_sock *llc = llc_sk(sk);
  347. struct llc_sap *sap = llc->sap;
  348. llc_pdu_header_init(skb, LLC_PDU_TYPE_I, sap->laddr.lsap,
  349. llc->daddr.lsap, LLC_PDU_CMD);
  350. llc_pdu_init_as_i_cmd(skb, 0, llc->vS, llc->vR);
  351. rc = llc_mac_hdr_init(skb, llc->dev->dev_addr, llc->daddr.mac);
  352. if (likely(!rc)) {
  353. llc_conn_send_pdu(sk, skb);
  354. llc_conn_ac_inc_vs_by_1(sk, skb);
  355. }
  356. return 0;
  357. }
  358. int llc_conn_ac_resend_i_xxx_x_set_0(struct sock *sk, struct sk_buff *skb)
  359. {
  360. struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
  361. u8 nr = LLC_I_GET_NR(pdu);
  362. llc_conn_resend_i_pdu_as_cmd(sk, nr, 0);
  363. return 0;
  364. }
  365. int llc_conn_ac_resend_i_xxx_x_set_0_or_send_rr(struct sock *sk,
  366. struct sk_buff *skb)
  367. {
  368. u8 nr;
  369. struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
  370. int rc = -ENOBUFS;
  371. struct llc_sock *llc = llc_sk(sk);
  372. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  373. if (nskb) {
  374. struct llc_sap *sap = llc->sap;
  375. llc_pdu_header_init(nskb, LLC_PDU_TYPE_U, sap->laddr.lsap,
  376. llc->daddr.lsap, LLC_PDU_RSP);
  377. llc_pdu_init_as_rr_rsp(nskb, 0, llc->vR);
  378. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  379. if (likely(!rc))
  380. llc_conn_send_pdu(sk, nskb);
  381. else
  382. kfree_skb(skb);
  383. }
  384. if (rc) {
  385. nr = LLC_I_GET_NR(pdu);
  386. rc = 0;
  387. llc_conn_resend_i_pdu_as_cmd(sk, nr, 0);
  388. }
  389. return rc;
  390. }
  391. int llc_conn_ac_resend_i_rsp_f_set_1(struct sock *sk, struct sk_buff *skb)
  392. {
  393. struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
  394. u8 nr = LLC_I_GET_NR(pdu);
  395. llc_conn_resend_i_pdu_as_rsp(sk, nr, 1);
  396. return 0;
  397. }
  398. int llc_conn_ac_send_rej_cmd_p_set_1(struct sock *sk, struct sk_buff *skb)
  399. {
  400. int rc = -ENOBUFS;
  401. struct llc_sock *llc = llc_sk(sk);
  402. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  403. if (nskb) {
  404. struct llc_sap *sap = llc->sap;
  405. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  406. llc->daddr.lsap, LLC_PDU_CMD);
  407. llc_pdu_init_as_rej_cmd(nskb, 1, llc->vR);
  408. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  409. if (unlikely(rc))
  410. goto free;
  411. llc_conn_send_pdu(sk, nskb);
  412. }
  413. out:
  414. return rc;
  415. free:
  416. kfree_skb(nskb);
  417. goto out;
  418. }
  419. int llc_conn_ac_send_rej_rsp_f_set_1(struct sock *sk, struct sk_buff *skb)
  420. {
  421. int rc = -ENOBUFS;
  422. struct llc_sock *llc = llc_sk(sk);
  423. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  424. if (nskb) {
  425. struct llc_sap *sap = llc->sap;
  426. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  427. llc->daddr.lsap, LLC_PDU_RSP);
  428. llc_pdu_init_as_rej_rsp(nskb, 1, llc->vR);
  429. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  430. if (unlikely(rc))
  431. goto free;
  432. llc_conn_send_pdu(sk, nskb);
  433. }
  434. out:
  435. return rc;
  436. free:
  437. kfree_skb(nskb);
  438. goto out;
  439. }
  440. int llc_conn_ac_send_rej_xxx_x_set_0(struct sock *sk, struct sk_buff *skb)
  441. {
  442. int rc = -ENOBUFS;
  443. struct llc_sock *llc = llc_sk(sk);
  444. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  445. if (nskb) {
  446. struct llc_sap *sap = llc->sap;
  447. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  448. llc->daddr.lsap, LLC_PDU_RSP);
  449. llc_pdu_init_as_rej_rsp(nskb, 0, llc->vR);
  450. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  451. if (unlikely(rc))
  452. goto free;
  453. llc_conn_send_pdu(sk, nskb);
  454. }
  455. out:
  456. return rc;
  457. free:
  458. kfree_skb(nskb);
  459. goto out;
  460. }
  461. int llc_conn_ac_send_rnr_cmd_p_set_1(struct sock *sk, struct sk_buff *skb)
  462. {
  463. int rc = -ENOBUFS;
  464. struct llc_sock *llc = llc_sk(sk);
  465. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  466. if (nskb) {
  467. struct llc_sap *sap = llc->sap;
  468. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  469. llc->daddr.lsap, LLC_PDU_CMD);
  470. llc_pdu_init_as_rnr_cmd(nskb, 1, llc->vR);
  471. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  472. if (unlikely(rc))
  473. goto free;
  474. llc_conn_send_pdu(sk, nskb);
  475. }
  476. out:
  477. return rc;
  478. free:
  479. kfree_skb(nskb);
  480. goto out;
  481. }
  482. int llc_conn_ac_send_rnr_rsp_f_set_1(struct sock *sk, struct sk_buff *skb)
  483. {
  484. int rc = -ENOBUFS;
  485. struct llc_sock *llc = llc_sk(sk);
  486. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  487. if (nskb) {
  488. struct llc_sap *sap = llc->sap;
  489. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  490. llc->daddr.lsap, LLC_PDU_RSP);
  491. llc_pdu_init_as_rnr_rsp(nskb, 1, llc->vR);
  492. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  493. if (unlikely(rc))
  494. goto free;
  495. llc_conn_send_pdu(sk, nskb);
  496. }
  497. out:
  498. return rc;
  499. free:
  500. kfree_skb(nskb);
  501. goto out;
  502. }
  503. int llc_conn_ac_send_rnr_xxx_x_set_0(struct sock *sk, struct sk_buff *skb)
  504. {
  505. int rc = -ENOBUFS;
  506. struct llc_sock *llc = llc_sk(sk);
  507. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  508. if (nskb) {
  509. struct llc_sap *sap = llc->sap;
  510. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  511. llc->daddr.lsap, LLC_PDU_RSP);
  512. llc_pdu_init_as_rnr_rsp(nskb, 0, llc->vR);
  513. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  514. if (unlikely(rc))
  515. goto free;
  516. llc_conn_send_pdu(sk, nskb);
  517. }
  518. out:
  519. return rc;
  520. free:
  521. kfree_skb(nskb);
  522. goto out;
  523. }
  524. int llc_conn_ac_set_remote_busy(struct sock *sk, struct sk_buff *skb)
  525. {
  526. struct llc_sock *llc = llc_sk(sk);
  527. if (!llc->remote_busy_flag) {
  528. llc->remote_busy_flag = 1;
  529. mod_timer(&llc->busy_state_timer.timer,
  530. jiffies + llc->busy_state_timer.expire);
  531. }
  532. return 0;
  533. }
  534. int llc_conn_ac_opt_send_rnr_xxx_x_set_0(struct sock *sk, struct sk_buff *skb)
  535. {
  536. int rc = -ENOBUFS;
  537. struct llc_sock *llc = llc_sk(sk);
  538. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  539. if (nskb) {
  540. struct llc_sap *sap = llc->sap;
  541. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  542. llc->daddr.lsap, LLC_PDU_RSP);
  543. llc_pdu_init_as_rnr_rsp(nskb, 0, llc->vR);
  544. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  545. if (unlikely(rc))
  546. goto free;
  547. llc_conn_send_pdu(sk, nskb);
  548. }
  549. out:
  550. return rc;
  551. free:
  552. kfree_skb(nskb);
  553. goto out;
  554. }
  555. int llc_conn_ac_send_rr_cmd_p_set_1(struct sock *sk, struct sk_buff *skb)
  556. {
  557. int rc = -ENOBUFS;
  558. struct llc_sock *llc = llc_sk(sk);
  559. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  560. if (nskb) {
  561. struct llc_sap *sap = llc->sap;
  562. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  563. llc->daddr.lsap, LLC_PDU_CMD);
  564. llc_pdu_init_as_rr_cmd(nskb, 1, llc->vR);
  565. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  566. if (unlikely(rc))
  567. goto free;
  568. llc_conn_send_pdu(sk, nskb);
  569. }
  570. out:
  571. return rc;
  572. free:
  573. kfree_skb(nskb);
  574. goto out;
  575. }
  576. int llc_conn_ac_send_rr_rsp_f_set_1(struct sock *sk, struct sk_buff *skb)
  577. {
  578. int rc = -ENOBUFS;
  579. struct llc_sock *llc = llc_sk(sk);
  580. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  581. if (nskb) {
  582. struct llc_sap *sap = llc->sap;
  583. u8 f_bit = 1;
  584. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  585. llc->daddr.lsap, LLC_PDU_RSP);
  586. llc_pdu_init_as_rr_rsp(nskb, f_bit, llc->vR);
  587. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  588. if (unlikely(rc))
  589. goto free;
  590. llc_conn_send_pdu(sk, nskb);
  591. }
  592. out:
  593. return rc;
  594. free:
  595. kfree_skb(nskb);
  596. goto out;
  597. }
  598. int llc_conn_ac_send_ack_rsp_f_set_1(struct sock *sk, struct sk_buff *skb)
  599. {
  600. int rc = -ENOBUFS;
  601. struct llc_sock *llc = llc_sk(sk);
  602. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  603. if (nskb) {
  604. struct llc_sap *sap = llc->sap;
  605. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  606. llc->daddr.lsap, LLC_PDU_RSP);
  607. llc_pdu_init_as_rr_rsp(nskb, 1, llc->vR);
  608. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  609. if (unlikely(rc))
  610. goto free;
  611. llc_conn_send_pdu(sk, nskb);
  612. }
  613. out:
  614. return rc;
  615. free:
  616. kfree_skb(nskb);
  617. goto out;
  618. }
  619. int llc_conn_ac_send_rr_xxx_x_set_0(struct sock *sk, struct sk_buff *skb)
  620. {
  621. int rc = -ENOBUFS;
  622. struct llc_sock *llc = llc_sk(sk);
  623. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  624. if (nskb) {
  625. struct llc_sap *sap = llc->sap;
  626. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  627. llc->daddr.lsap, LLC_PDU_RSP);
  628. llc_pdu_init_as_rr_rsp(nskb, 0, llc->vR);
  629. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  630. if (unlikely(rc))
  631. goto free;
  632. llc_conn_send_pdu(sk, nskb);
  633. }
  634. out:
  635. return rc;
  636. free:
  637. kfree_skb(nskb);
  638. goto out;
  639. }
  640. int llc_conn_ac_send_ack_xxx_x_set_0(struct sock *sk, struct sk_buff *skb)
  641. {
  642. int rc = -ENOBUFS;
  643. struct llc_sock *llc = llc_sk(sk);
  644. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  645. if (nskb) {
  646. struct llc_sap *sap = llc->sap;
  647. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  648. llc->daddr.lsap, LLC_PDU_RSP);
  649. llc_pdu_init_as_rr_rsp(nskb, 0, llc->vR);
  650. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  651. if (unlikely(rc))
  652. goto free;
  653. llc_conn_send_pdu(sk, nskb);
  654. }
  655. out:
  656. return rc;
  657. free:
  658. kfree_skb(nskb);
  659. goto out;
  660. }
  661. void llc_conn_set_p_flag(struct sock *sk, u8 value)
  662. {
  663. int state_changed = llc_sk(sk)->p_flag && !value;
  664. llc_sk(sk)->p_flag = value;
  665. if (state_changed)
  666. sk->sk_state_change(sk);
  667. }
  668. int llc_conn_ac_send_sabme_cmd_p_set_x(struct sock *sk, struct sk_buff *skb)
  669. {
  670. int rc = -ENOBUFS;
  671. struct llc_sock *llc = llc_sk(sk);
  672. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  673. if (nskb) {
  674. struct llc_sap *sap = llc->sap;
  675. u8 *dmac = llc->daddr.mac;
  676. if (llc->dev->flags & IFF_LOOPBACK)
  677. dmac = llc->dev->dev_addr;
  678. llc_pdu_header_init(nskb, LLC_PDU_TYPE_U, sap->laddr.lsap,
  679. llc->daddr.lsap, LLC_PDU_CMD);
  680. llc_pdu_init_as_sabme_cmd(nskb, 1);
  681. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, dmac);
  682. if (unlikely(rc))
  683. goto free;
  684. llc_conn_send_pdu(sk, nskb);
  685. llc_conn_set_p_flag(sk, 1);
  686. }
  687. out:
  688. return rc;
  689. free:
  690. kfree_skb(nskb);
  691. goto out;
  692. }
  693. int llc_conn_ac_send_ua_rsp_f_set_p(struct sock *sk, struct sk_buff *skb)
  694. {
  695. u8 f_bit;
  696. int rc = -ENOBUFS;
  697. struct llc_sock *llc = llc_sk(sk);
  698. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  699. llc_pdu_decode_pf_bit(skb, &f_bit);
  700. if (nskb) {
  701. struct llc_sap *sap = llc->sap;
  702. nskb->dev = llc->dev;
  703. llc_pdu_header_init(nskb, LLC_PDU_TYPE_U, sap->laddr.lsap,
  704. llc->daddr.lsap, LLC_PDU_RSP);
  705. llc_pdu_init_as_ua_rsp(nskb, f_bit);
  706. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  707. if (unlikely(rc))
  708. goto free;
  709. llc_conn_send_pdu(sk, nskb);
  710. }
  711. out:
  712. return rc;
  713. free:
  714. kfree_skb(nskb);
  715. goto out;
  716. }
  717. int llc_conn_ac_set_s_flag_0(struct sock *sk, struct sk_buff *skb)
  718. {
  719. llc_sk(sk)->s_flag = 0;
  720. return 0;
  721. }
  722. int llc_conn_ac_set_s_flag_1(struct sock *sk, struct sk_buff *skb)
  723. {
  724. llc_sk(sk)->s_flag = 1;
  725. return 0;
  726. }
  727. int llc_conn_ac_start_p_timer(struct sock *sk, struct sk_buff *skb)
  728. {
  729. struct llc_sock *llc = llc_sk(sk);
  730. llc_conn_set_p_flag(sk, 1);
  731. mod_timer(&llc->pf_cycle_timer.timer,
  732. jiffies + llc->pf_cycle_timer.expire);
  733. return 0;
  734. }
  735. /**
  736. * llc_conn_ac_send_ack_if_needed - check if ack is needed
  737. * @sk: current connection structure
  738. * @skb: current event
  739. *
  740. * Checks number of received PDUs which have not been acknowledged, yet,
  741. * If number of them reaches to "npta"(Number of PDUs To Acknowledge) then
  742. * sends an RR response as acknowledgement for them. Returns 0 for
  743. * success, 1 otherwise.
  744. */
  745. int llc_conn_ac_send_ack_if_needed(struct sock *sk, struct sk_buff *skb)
  746. {
  747. u8 pf_bit;
  748. struct llc_sock *llc = llc_sk(sk);
  749. llc_pdu_decode_pf_bit(skb, &pf_bit);
  750. llc->ack_pf |= pf_bit & 1;
  751. if (!llc->ack_must_be_send) {
  752. llc->first_pdu_Ns = llc->vR;
  753. llc->ack_must_be_send = 1;
  754. llc->ack_pf = pf_bit & 1;
  755. }
  756. if (((llc->vR - llc->first_pdu_Ns + 129) % 128) >= llc->npta) {
  757. llc_conn_ac_send_rr_rsp_f_set_ackpf(sk, skb);
  758. llc->ack_must_be_send = 0;
  759. llc->ack_pf = 0;
  760. llc_conn_ac_inc_npta_value(sk, skb);
  761. }
  762. return 0;
  763. }
  764. /**
  765. * llc_conn_ac_rst_sendack_flag - resets ack_must_be_send flag
  766. * @sk: current connection structure
  767. * @skb: current event
  768. *
  769. * This action resets ack_must_be_send flag of given connection, this flag
  770. * indicates if there is any PDU which has not been acknowledged yet.
  771. * Returns 0 for success, 1 otherwise.
  772. */
  773. int llc_conn_ac_rst_sendack_flag(struct sock *sk, struct sk_buff *skb)
  774. {
  775. llc_sk(sk)->ack_must_be_send = llc_sk(sk)->ack_pf = 0;
  776. return 0;
  777. }
  778. /**
  779. * llc_conn_ac_send_i_rsp_f_set_ackpf - acknowledge received PDUs
  780. * @sk: current connection structure
  781. * @skb: current event
  782. *
  783. * Sends an I response PDU with f-bit set to ack_pf flag as acknowledge to
  784. * all received PDUs which have not been acknowledged, yet. ack_pf flag is
  785. * set to one if one PDU with p-bit set to one is received. Returns 0 for
  786. * success, 1 otherwise.
  787. */
  788. static int llc_conn_ac_send_i_rsp_f_set_ackpf(struct sock *sk,
  789. struct sk_buff *skb)
  790. {
  791. int rc;
  792. struct llc_sock *llc = llc_sk(sk);
  793. struct llc_sap *sap = llc->sap;
  794. llc_pdu_header_init(skb, LLC_PDU_TYPE_I, sap->laddr.lsap,
  795. llc->daddr.lsap, LLC_PDU_RSP);
  796. llc_pdu_init_as_i_cmd(skb, llc->ack_pf, llc->vS, llc->vR);
  797. rc = llc_mac_hdr_init(skb, llc->dev->dev_addr, llc->daddr.mac);
  798. if (likely(!rc)) {
  799. llc_conn_send_pdu(sk, skb);
  800. llc_conn_ac_inc_vs_by_1(sk, skb);
  801. }
  802. return rc;
  803. }
  804. /**
  805. * llc_conn_ac_send_i_as_ack - sends an I-format PDU to acknowledge rx PDUs
  806. * @sk: current connection structure.
  807. * @skb: current event.
  808. *
  809. * This action sends an I-format PDU as acknowledge to received PDUs which
  810. * have not been acknowledged, yet, if there is any. By using of this
  811. * action number of acknowledgements decreases, this technic is called
  812. * piggy backing. Returns 0 for success, 1 otherwise.
  813. */
  814. int llc_conn_ac_send_i_as_ack(struct sock *sk, struct sk_buff *skb)
  815. {
  816. struct llc_sock *llc = llc_sk(sk);
  817. if (llc->ack_must_be_send) {
  818. llc_conn_ac_send_i_rsp_f_set_ackpf(sk, skb);
  819. llc->ack_must_be_send = 0 ;
  820. llc->ack_pf = 0;
  821. } else
  822. llc_conn_ac_send_i_cmd_p_set_0(sk, skb);
  823. return 0;
  824. }
  825. /**
  826. * llc_conn_ac_send_rr_rsp_f_set_ackpf - ack all rx PDUs not yet acked
  827. * @sk: current connection structure.
  828. * @skb: current event.
  829. *
  830. * This action sends an RR response with f-bit set to ack_pf flag as
  831. * acknowledge to all received PDUs which have not been acknowledged, yet,
  832. * if there is any. ack_pf flag indicates if a PDU has been received with
  833. * p-bit set to one. Returns 0 for success, 1 otherwise.
  834. */
  835. static int llc_conn_ac_send_rr_rsp_f_set_ackpf(struct sock *sk,
  836. struct sk_buff *skb)
  837. {
  838. int rc = -ENOBUFS;
  839. struct llc_sock *llc = llc_sk(sk);
  840. struct sk_buff *nskb = llc_alloc_frame(sk, llc->dev);
  841. if (nskb) {
  842. struct llc_sap *sap = llc->sap;
  843. llc_pdu_header_init(nskb, LLC_PDU_TYPE_S, sap->laddr.lsap,
  844. llc->daddr.lsap, LLC_PDU_RSP);
  845. llc_pdu_init_as_rr_rsp(nskb, llc->ack_pf, llc->vR);
  846. rc = llc_mac_hdr_init(nskb, llc->dev->dev_addr, llc->daddr.mac);
  847. if (unlikely(rc))
  848. goto free;
  849. llc_conn_send_pdu(sk, nskb);
  850. }
  851. out:
  852. return rc;
  853. free:
  854. kfree_skb(nskb);
  855. goto out;
  856. }
  857. /**
  858. * llc_conn_ac_inc_npta_value - tries to make value of npta greater
  859. * @sk: current connection structure.
  860. * @skb: current event.
  861. *
  862. * After "inc_cntr" times calling of this action, "npta" increase by one.
  863. * this action tries to make vale of "npta" greater as possible; number of
  864. * acknowledgements decreases by increasing of "npta". Returns 0 for
  865. * success, 1 otherwise.
  866. */
  867. static int llc_conn_ac_inc_npta_value(struct sock *sk, struct sk_buff *skb)
  868. {
  869. struct llc_sock *llc = llc_sk(sk);
  870. if (!llc->inc_cntr) {
  871. llc->dec_step = 0;
  872. llc->dec_cntr = llc->inc_cntr = 2;
  873. ++llc->npta;
  874. if (llc->npta > 127)
  875. llc->npta = 127 ;
  876. } else
  877. --llc->inc_cntr;
  878. return 0;
  879. }
  880. /**
  881. * llc_conn_ac_adjust_npta_by_rr - decreases "npta" by one
  882. * @sk: current connection structure.
  883. * @skb: current event.
  884. *
  885. * After receiving "dec_cntr" times RR command, this action decreases
  886. * "npta" by one. Returns 0 for success, 1 otherwise.
  887. */
  888. int llc_conn_ac_adjust_npta_by_rr(struct sock *sk, struct sk_buff *skb)
  889. {
  890. struct llc_sock *llc = llc_sk(sk);
  891. if (!llc->connect_step && !llc->remote_busy_flag) {
  892. if (!llc->dec_step) {
  893. if (!llc->dec_cntr) {
  894. llc->inc_cntr = llc->dec_cntr = 2;
  895. if (llc->npta > 0)
  896. llc->npta = llc->npta - 1;
  897. } else
  898. llc->dec_cntr -=1;
  899. }
  900. } else
  901. llc->connect_step = 0 ;
  902. return 0;
  903. }
  904. /**
  905. * llc_conn_ac_adjust_npta_by_rnr - decreases "npta" by one
  906. * @sk: current connection structure.
  907. * @skb: current event.
  908. *
  909. * After receiving "dec_cntr" times RNR command, this action decreases
  910. * "npta" by one. Returns 0 for success, 1 otherwise.
  911. */
  912. int llc_conn_ac_adjust_npta_by_rnr(struct sock *sk, struct sk_buff *skb)
  913. {
  914. struct llc_sock *llc = llc_sk(sk);
  915. if (llc->remote_busy_flag)
  916. if (!llc->dec_step) {
  917. if (!llc->dec_cntr) {
  918. llc->inc_cntr = llc->dec_cntr = 2;
  919. if (llc->npta > 0)
  920. --llc->npta;
  921. } else
  922. --llc->dec_cntr;
  923. }
  924. return 0;
  925. }
  926. /**
  927. * llc_conn_ac_dec_tx_win_size - decreases tx window size
  928. * @sk: current connection structure.
  929. * @skb: current event.
  930. *
  931. * After receiving of a REJ command or response, transmit window size is
  932. * decreased by number of PDUs which are outstanding yet. Returns 0 for
  933. * success, 1 otherwise.
  934. */
  935. int llc_conn_ac_dec_tx_win_size(struct sock *sk, struct sk_buff *skb)
  936. {
  937. struct llc_sock *llc = llc_sk(sk);
  938. u8 unacked_pdu = skb_queue_len(&llc->pdu_unack_q);
  939. llc->k -= unacked_pdu;
  940. if (llc->k < 2)
  941. llc->k = 2;
  942. return 0;
  943. }
  944. /**
  945. * llc_conn_ac_inc_tx_win_size - tx window size is inc by 1
  946. * @sk: current connection structure.
  947. * @skb: current event.
  948. *
  949. * After receiving an RR response with f-bit set to one, transmit window
  950. * size is increased by one. Returns 0 for success, 1 otherwise.
  951. */
  952. int llc_conn_ac_inc_tx_win_size(struct sock *sk, struct sk_buff *skb)
  953. {
  954. struct llc_sock *llc = llc_sk(sk);
  955. llc->k += 1;
  956. if (llc->k > 128)
  957. llc->k = 128 ;
  958. return 0;
  959. }
  960. int llc_conn_ac_stop_all_timers(struct sock *sk, struct sk_buff *skb)
  961. {
  962. struct llc_sock *llc = llc_sk(sk);
  963. del_timer(&llc->pf_cycle_timer.timer);
  964. del_timer(&llc->ack_timer.timer);
  965. del_timer(&llc->rej_sent_timer.timer);
  966. del_timer(&llc->busy_state_timer.timer);
  967. llc->ack_must_be_send = 0;
  968. llc->ack_pf = 0;
  969. return 0;
  970. }
  971. int llc_conn_ac_stop_other_timers(struct sock *sk, struct sk_buff *skb)
  972. {
  973. struct llc_sock *llc = llc_sk(sk);
  974. del_timer(&llc->rej_sent_timer.timer);
  975. del_timer(&llc->pf_cycle_timer.timer);
  976. del_timer(&llc->busy_state_timer.timer);
  977. llc->ack_must_be_send = 0;
  978. llc->ack_pf = 0;
  979. return 0;
  980. }
  981. int llc_conn_ac_start_ack_timer(struct sock *sk, struct sk_buff *skb)
  982. {
  983. struct llc_sock *llc = llc_sk(sk);
  984. mod_timer(&llc->ack_timer.timer, jiffies + llc->ack_timer.expire);
  985. return 0;
  986. }
  987. int llc_conn_ac_start_rej_timer(struct sock *sk, struct sk_buff *skb)
  988. {
  989. struct llc_sock *llc = llc_sk(sk);
  990. mod_timer(&llc->rej_sent_timer.timer,
  991. jiffies + llc->rej_sent_timer.expire);
  992. return 0;
  993. }
  994. int llc_conn_ac_start_ack_tmr_if_not_running(struct sock *sk,
  995. struct sk_buff *skb)
  996. {
  997. struct llc_sock *llc = llc_sk(sk);
  998. if (!timer_pending(&llc->ack_timer.timer))
  999. mod_timer(&llc->ack_timer.timer,
  1000. jiffies + llc->ack_timer.expire);
  1001. return 0;
  1002. }
  1003. int llc_conn_ac_stop_ack_timer(struct sock *sk, struct sk_buff *skb)
  1004. {
  1005. del_timer(&llc_sk(sk)->ack_timer.timer);
  1006. return 0;
  1007. }
  1008. int llc_conn_ac_stop_p_timer(struct sock *sk, struct sk_buff *skb)
  1009. {
  1010. struct llc_sock *llc = llc_sk(sk);
  1011. del_timer(&llc->pf_cycle_timer.timer);
  1012. llc_conn_set_p_flag(sk, 0);
  1013. return 0;
  1014. }
  1015. int llc_conn_ac_stop_rej_timer(struct sock *sk, struct sk_buff *skb)
  1016. {
  1017. del_timer(&llc_sk(sk)->rej_sent_timer.timer);
  1018. return 0;
  1019. }
  1020. int llc_conn_ac_upd_nr_received(struct sock *sk, struct sk_buff *skb)
  1021. {
  1022. int acked;
  1023. u16 unacked = 0;
  1024. struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
  1025. struct llc_sock *llc = llc_sk(sk);
  1026. llc->last_nr = PDU_SUPV_GET_Nr(pdu);
  1027. acked = llc_conn_remove_acked_pdus(sk, llc->last_nr, &unacked);
  1028. /* On loopback we don't queue I frames in unack_pdu_q queue. */
  1029. if (acked > 0 || (llc->dev->flags & IFF_LOOPBACK)) {
  1030. llc->retry_count = 0;
  1031. del_timer(&llc->ack_timer.timer);
  1032. if (llc->failed_data_req) {
  1033. /* already, we did not accept data from upper layer
  1034. * (tx_window full or unacceptable state). Now, we
  1035. * can send data and must inform to upper layer.
  1036. */
  1037. llc->failed_data_req = 0;
  1038. llc_conn_ac_data_confirm(sk, skb);
  1039. }
  1040. if (unacked)
  1041. mod_timer(&llc->ack_timer.timer,
  1042. jiffies + llc->ack_timer.expire);
  1043. } else if (llc->failed_data_req) {
  1044. u8 f_bit;
  1045. llc_pdu_decode_pf_bit(skb, &f_bit);
  1046. if (f_bit == 1) {
  1047. llc->failed_data_req = 0;
  1048. llc_conn_ac_data_confirm(sk, skb);
  1049. }
  1050. }
  1051. return 0;
  1052. }
  1053. int llc_conn_ac_upd_p_flag(struct sock *sk, struct sk_buff *skb)
  1054. {
  1055. struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
  1056. if (LLC_PDU_IS_RSP(pdu)) {
  1057. u8 f_bit;
  1058. llc_pdu_decode_pf_bit(skb, &f_bit);
  1059. if (f_bit) {
  1060. llc_conn_set_p_flag(sk, 0);
  1061. llc_conn_ac_stop_p_timer(sk, skb);
  1062. }
  1063. }
  1064. return 0;
  1065. }
  1066. int llc_conn_ac_set_data_flag_2(struct sock *sk, struct sk_buff *skb)
  1067. {
  1068. llc_sk(sk)->data_flag = 2;
  1069. return 0;
  1070. }
  1071. int llc_conn_ac_set_data_flag_0(struct sock *sk, struct sk_buff *skb)
  1072. {
  1073. llc_sk(sk)->data_flag = 0;
  1074. return 0;
  1075. }
  1076. int llc_conn_ac_set_data_flag_1(struct sock *sk, struct sk_buff *skb)
  1077. {
  1078. llc_sk(sk)->data_flag = 1;
  1079. return 0;
  1080. }
  1081. int llc_conn_ac_set_data_flag_1_if_data_flag_eq_0(struct sock *sk,
  1082. struct sk_buff *skb)
  1083. {
  1084. if (!llc_sk(sk)->data_flag)
  1085. llc_sk(sk)->data_flag = 1;
  1086. return 0;
  1087. }
  1088. int llc_conn_ac_set_p_flag_0(struct sock *sk, struct sk_buff *skb)
  1089. {
  1090. llc_conn_set_p_flag(sk, 0);
  1091. return 0;
  1092. }
  1093. static int llc_conn_ac_set_p_flag_1(struct sock *sk, struct sk_buff *skb)
  1094. {
  1095. llc_conn_set_p_flag(sk, 1);
  1096. return 0;
  1097. }
  1098. int llc_conn_ac_set_remote_busy_0(struct sock *sk, struct sk_buff *skb)
  1099. {
  1100. llc_sk(sk)->remote_busy_flag = 0;
  1101. return 0;
  1102. }
  1103. int llc_conn_ac_set_cause_flag_0(struct sock *sk, struct sk_buff *skb)
  1104. {
  1105. llc_sk(sk)->cause_flag = 0;
  1106. return 0;
  1107. }
  1108. int llc_conn_ac_set_cause_flag_1(struct sock *sk, struct sk_buff *skb)
  1109. {
  1110. llc_sk(sk)->cause_flag = 1;
  1111. return 0;
  1112. }
  1113. int llc_conn_ac_set_retry_cnt_0(struct sock *sk, struct sk_buff *skb)
  1114. {
  1115. llc_sk(sk)->retry_count = 0;
  1116. return 0;
  1117. }
  1118. int llc_conn_ac_inc_retry_cnt_by_1(struct sock *sk, struct sk_buff *skb)
  1119. {
  1120. llc_sk(sk)->retry_count++;
  1121. return 0;
  1122. }
  1123. int llc_conn_ac_set_vr_0(struct sock *sk, struct sk_buff *skb)
  1124. {
  1125. llc_sk(sk)->vR = 0;
  1126. return 0;
  1127. }
  1128. int llc_conn_ac_inc_vr_by_1(struct sock *sk, struct sk_buff *skb)
  1129. {
  1130. llc_sk(sk)->vR = PDU_GET_NEXT_Vr(llc_sk(sk)->vR);
  1131. return 0;
  1132. }
  1133. int llc_conn_ac_set_vs_0(struct sock *sk, struct sk_buff *skb)
  1134. {
  1135. llc_sk(sk)->vS = 0;
  1136. return 0;
  1137. }
  1138. int llc_conn_ac_set_vs_nr(struct sock *sk, struct sk_buff *skb)
  1139. {
  1140. llc_sk(sk)->vS = llc_sk(sk)->last_nr;
  1141. return 0;
  1142. }
  1143. static int llc_conn_ac_inc_vs_by_1(struct sock *sk, struct sk_buff *skb)
  1144. {
  1145. llc_sk(sk)->vS = (llc_sk(sk)->vS + 1) % 128;
  1146. return 0;
  1147. }
  1148. static void llc_conn_tmr_common_cb(unsigned long timeout_data, u8 type)
  1149. {
  1150. struct sock *sk = (struct sock *)timeout_data;
  1151. struct sk_buff *skb = alloc_skb(0, GFP_ATOMIC);
  1152. bh_lock_sock(sk);
  1153. if (skb) {
  1154. struct llc_conn_state_ev *ev = llc_conn_ev(skb);
  1155. skb_set_owner_r(skb, sk);
  1156. ev->type = type;
  1157. llc_process_tmr_ev(sk, skb);
  1158. }
  1159. bh_unlock_sock(sk);
  1160. }
  1161. void llc_conn_pf_cycle_tmr_cb(unsigned long timeout_data)
  1162. {
  1163. llc_conn_tmr_common_cb(timeout_data, LLC_CONN_EV_TYPE_P_TMR);
  1164. }
  1165. void llc_conn_busy_tmr_cb(unsigned long timeout_data)
  1166. {
  1167. llc_conn_tmr_common_cb(timeout_data, LLC_CONN_EV_TYPE_BUSY_TMR);
  1168. }
  1169. void llc_conn_ack_tmr_cb(unsigned long timeout_data)
  1170. {
  1171. llc_conn_tmr_common_cb(timeout_data, LLC_CONN_EV_TYPE_ACK_TMR);
  1172. }
  1173. void llc_conn_rej_tmr_cb(unsigned long timeout_data)
  1174. {
  1175. llc_conn_tmr_common_cb(timeout_data, LLC_CONN_EV_TYPE_REJ_TMR);
  1176. }
  1177. int llc_conn_ac_rst_vs(struct sock *sk, struct sk_buff *skb)
  1178. {
  1179. llc_sk(sk)->X = llc_sk(sk)->vS;
  1180. llc_conn_ac_set_vs_nr(sk, skb);
  1181. return 0;
  1182. }
  1183. int llc_conn_ac_upd_vs(struct sock *sk, struct sk_buff *skb)
  1184. {
  1185. struct llc_pdu_sn *pdu = llc_pdu_sn_hdr(skb);
  1186. u8 nr = PDU_SUPV_GET_Nr(pdu);
  1187. if (llc_circular_between(llc_sk(sk)->vS, nr, llc_sk(sk)->X))
  1188. llc_conn_ac_set_vs_nr(sk, skb);
  1189. return 0;
  1190. }
  1191. /*
  1192. * Non-standard actions; these not contained in IEEE specification; for
  1193. * our own usage
  1194. */
  1195. /**
  1196. * llc_conn_disc - removes connection from SAP list and frees it
  1197. * @sk: closed connection
  1198. * @skb: occurred event
  1199. */
  1200. int llc_conn_disc(struct sock *sk, struct sk_buff *skb)
  1201. {
  1202. /* FIXME: this thing seems to want to die */
  1203. return 0;
  1204. }
  1205. /**
  1206. * llc_conn_reset - resets connection
  1207. * @sk : reseting connection.
  1208. * @skb: occurred event.
  1209. *
  1210. * Stop all timers, empty all queues and reset all flags.
  1211. */
  1212. int llc_conn_reset(struct sock *sk, struct sk_buff *skb)
  1213. {
  1214. llc_sk_reset(sk);
  1215. return 0;
  1216. }
  1217. /**
  1218. * llc_circular_between - designates that b is between a and c or not
  1219. * @a: lower bound
  1220. * @b: element to see if is between a and b
  1221. * @c: upper bound
  1222. *
  1223. * This function designates that b is between a and c or not (for example,
  1224. * 0 is between 127 and 1). Returns 1 if b is between a and c, 0
  1225. * otherwise.
  1226. */
  1227. u8 llc_circular_between(u8 a, u8 b, u8 c)
  1228. {
  1229. b = b - a;
  1230. c = c - a;
  1231. return b <= c;
  1232. }
  1233. /**
  1234. * llc_process_tmr_ev - timer backend
  1235. * @sk: active connection
  1236. * @skb: occurred event
  1237. *
  1238. * This function is called from timer callback functions. When connection
  1239. * is busy (during sending a data frame) timer expiration event must be
  1240. * queued. Otherwise this event can be sent to connection state machine.
  1241. * Queued events will process by llc_backlog_rcv function after sending
  1242. * data frame.
  1243. */
  1244. static void llc_process_tmr_ev(struct sock *sk, struct sk_buff *skb)
  1245. {
  1246. if (llc_sk(sk)->state == LLC_CONN_OUT_OF_SVC) {
  1247. printk(KERN_WARNING "%s: timer called on closed connection\n",
  1248. __FUNCTION__);
  1249. kfree_skb(skb);
  1250. } else {
  1251. if (!sock_owned_by_user(sk))
  1252. llc_conn_state_process(sk, skb);
  1253. else {
  1254. llc_set_backlog_type(skb, LLC_EVENT);
  1255. sk_add_backlog(sk, skb);
  1256. }
  1257. }
  1258. }